1
/* Copyright (C) 2000 MySQL AB
3
This program is free software; you can redistribute it and/or modify
4
it under the terms of the GNU General Public License as published by
5
the Free Software Foundation; version 2 of the License.
7
This program is distributed in the hope that it will be useful,
8
but WITHOUT ANY WARRANTY; without even the implied warranty of
9
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10
GNU General Public License for more details.
12
You should have received a copy of the GNU General Public License
13
along with this program; if not, write to the Free Software
14
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
1
/* -*- mode: c++; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
* vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
4
* Copyright (C) 2008 Sun Microsystems, Inc.
6
* This program is free software; you can redistribute it and/or modify
7
* it under the terms of the GNU General Public License as published by
8
* the Free Software Foundation; version 2 of the License.
10
* This program is distributed in the hope that it will be useful,
11
* but WITHOUT ANY WARRANTY; without even the implied warranty of
12
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13
* GNU General Public License for more details.
15
* You should have received a copy of the GNU General Public License
16
* along with this program; if not, write to the Free Software
17
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
20
#ifndef DRIZZLED_ITEM_SUBSELECT_H
21
#define DRIZZLED_ITEM_SUBSELECT_H
16
23
/* subselect Item */
18
#ifdef USE_PRAGMA_INTERFACE
19
#pragma interface /* gcc class implementation */
23
class st_select_lex_unit;
26
#include "drizzled/comp_creator.h"
27
#include "drizzled/item/ref.h"
28
#include "drizzled/item/field.h"
29
#include "drizzled/item/bin_string.h"
30
#include "drizzled/util/test.h"
36
class Select_Lex_Unit;
25
38
class select_result_interceptor;
26
39
class subselect_engine;
27
40
class subselect_hash_sj_engine;
28
41
class Item_bool_func2;
43
class Item_in_optimizer;
44
class Item_func_not_all;
45
class Tmp_Table_Param;
31
48
/* base class for subselects */
35
52
bool value_assigned; /* value already assigned to subselect */
37
54
/* thread handler, will be assigned in fix_fields only */
39
56
/* substitution instead of subselect in case of optimization */
40
57
Item *substitution;
41
58
/* unit of subquery */
43
st_select_lex_unit *unit;
59
Select_Lex_Unit *unit;
45
61
/* engine that perform execution of subselect (single select or union) */
46
62
subselect_engine *engine;
49
65
/* cache of used external tables */
50
66
table_map used_tables_cache;
51
67
/* allowed number of columns (1 for single value subqueries) */
53
69
/* where subquery is placed */
54
70
enum_parsing_place parsing_place;
55
71
/* work with 'substitution' */
79
95
pointer in constructor initialization list, but we need to pass a pointer
80
96
to subselect Item class to select_result_interceptor's constructor.
82
virtual void init (st_select_lex *select_lex,
98
virtual void init (Select_Lex *select_lex,
83
99
select_result_interceptor *result);
85
101
~Item_subselect();
105
121
bool const_item() const;
106
122
inline table_map get_used_tables_cache() { return used_tables_cache; }
107
123
inline bool get_const_item_cache() { return const_item_cache; }
108
Item *get_tmp_table_item(THD *thd);
124
Item *get_tmp_table_item(Session *session);
109
125
void update_used_tables();
110
126
virtual void print(String *str, enum_query_type query_type);
111
127
virtual bool have_guarded_conds() { return false; }
130
146
virtual void reset_value_registration() {}
131
147
enum_parsing_place place() { return parsing_place; }
132
bool walk(Item_processor processor, bool walk_subquery, uchar *arg);
148
bool walk(Item_processor processor, bool walk_subquery, unsigned char *arg);
135
Get the SELECT_LEX structure associated with this Item.
136
@return the SELECT_LEX structure associated with this Item
151
Get the Select_Lex structure associated with this Item.
152
@return the Select_Lex structure associated with this Item
138
st_select_lex* get_select_lex();
154
Select_Lex* get_select_lex();
140
156
friend class select_result_interceptor;
141
157
friend class Item_in_optimizer;
142
friend bool Item_field::fix_fields(THD *, Item **);
143
friend int Item_field::fix_outer_field(THD *, Field **, Item **);
144
friend bool Item_ref::fix_fields(THD *, Item **);
145
friend void mark_select_range_as_dependent(THD*,
146
st_select_lex*, st_select_lex*,
158
friend bool Item_field::fix_fields(Session *, Item **);
159
friend int Item_field::fix_outer_field(Session *, Field **, Item **);
160
friend bool Item_ref::fix_fields(Session *, Item **);
161
friend void mark_select_range_as_dependent(Session*,
162
Select_Lex*, Select_Lex*,
147
163
Field*, Item*, Item_ident*);
156
172
Item_cache *value, **row;
158
Item_singlerow_subselect(st_select_lex *select_lex);
174
Item_singlerow_subselect(Select_Lex *select_lex);
159
175
Item_singlerow_subselect() :Item_subselect(), value(0), row (0) {}
162
178
subs_type substype() { return SINGLEROW_SUBS; }
165
trans_res select_transformer(JOIN *join);
166
void store(uint i, Item* item);
181
trans_res select_transformer(Join *join);
182
void store(uint32_t i, Item* item);
167
183
double val_real();
168
184
int64_t val_int ();
169
185
String *val_str (String *);
173
189
enum_field_types field_type() const;
174
190
void fix_length_and_dec();
177
Item* element_index(uint i) { return my_reinterpret_cast(Item*)(row[i]); }
178
Item** addr(uint i) { return (Item**)row + i; }
179
bool check_cols(uint c);
193
Item* element_index(uint32_t i) { return reinterpret_cast<Item*>(row[i]); }
194
Item** addr(uint32_t i) { return (Item**)row + i; }
195
bool check_cols(uint32_t c);
180
196
bool null_inside();
181
197
void bring_value();
186
202
The only caller of this method is handle_sql2003_note184_exception(),
187
203
see the code there for more details.
188
204
Note that this method breaks the object internal integrity, by
189
removing it's association with the corresponding SELECT_LEX,
205
removing it's association with the corresponding Select_Lex,
190
206
making this object orphan from the parse tree.
191
207
No other method, beside the destructor, should be called on this
192
208
object, as it is now invalid.
193
@return the SELECT_LEX structure that was given in the constructor.
209
@return the Select_Lex structure that was given in the constructor.
195
st_select_lex* invalidate_and_restore_select_lex();
211
Select_Lex* invalidate_and_restore_select_lex();
197
213
friend class select_singlerow_subselect;
206
222
bool was_values; // Set if we have found at least one row
208
Item_maxmin_subselect(THD *thd, Item_subselect *parent,
209
st_select_lex *select_lex, bool max);
224
Item_maxmin_subselect(Session *session, Item_subselect *parent,
225
Select_Lex *select_lex, bool max);
210
226
virtual void print(String *str, enum_query_type query_type);
212
228
bool any_value() { return was_values; }
286
302
/* Used to trigger on/off conditions that were pushed down to subselect */
287
303
bool *pushed_cond_guards;
289
305
/* Priority of this predicate in the convert-to-semi-join-nest process. */
290
306
int sj_convert_priority;
293
309
Location of the subquery predicate. It is either
294
310
- pointer to join nest if the subquery predicate is in the ON expression
295
- (TABLE_LIST*)1 if the predicate is in the WHERE.
311
- (TableList*)1 if the predicate is in the WHERE.
297
TABLE_LIST *expr_join_nest;
313
TableList *expr_join_nest;
299
315
/* The method chosen to execute the IN predicate. */
300
316
enum enum_exec_method {
319
335
Item_func_not_all *upper_item; // point on NOT/NOP before ALL/SOME subquery
321
Item_in_subselect(Item * left_expr, st_select_lex *select_lex);
337
Item_in_subselect(Item * left_expr, Select_Lex *select_lex);
322
338
Item_in_subselect()
323
:Item_exists_subselect(), left_expr_cache(0), first_execution(true),
324
optimizer(0), abort_on_null(0), pushed_cond_guards(NULL),
325
exec_method(NOT_TRANSFORMED), upper_item(0)
340
Item_exists_subselect(),
342
left_expr_cache(NULL),
343
first_execution(true),
345
abort_on_null(false),
346
pushed_cond_guards(NULL),
347
sj_convert_priority(0),
348
expr_join_nest(NULL),
349
exec_method(NOT_TRANSFORMED),
328
353
subs_type substype() { return IN_SUBS; }
335
trans_res select_transformer(JOIN *join);
336
trans_res select_in_like_transformer(JOIN *join, Comp_creator *func);
337
trans_res single_value_transformer(JOIN *join, Comp_creator *func);
338
trans_res row_value_transformer(JOIN * join);
339
trans_res single_value_in_to_exists_transformer(JOIN * join,
341
trans_res row_value_in_to_exists_transformer(JOIN * join);
360
trans_res select_transformer(Join *join);
361
trans_res select_in_like_transformer(Join *join, const Comp_creator *func);
362
trans_res single_value_transformer(Join *join, const Comp_creator *func);
363
trans_res row_value_transformer(Join * join);
364
trans_res single_value_in_to_exists_transformer(Join * join,
365
const Comp_creator *func);
366
trans_res row_value_in_to_exists_transformer(Join * join);
342
367
virtual bool exec();
343
368
int64_t val_int();
344
369
double val_real();
349
374
void top_level_item() { abort_on_null=1; }
350
375
inline bool is_top_level_item() { return abort_on_null; }
351
bool test_limit(st_select_lex_unit *unit);
376
bool test_limit(Select_Lex_Unit *unit);
352
377
virtual void print(String *str, enum_query_type query_type);
353
bool fix_fields(THD *thd, Item **ref);
378
bool fix_fields(Session *session, Item **ref);
354
379
bool setup_engine();
355
380
bool init_left_expr_cache();
356
bool is_expensive_processor(uchar *arg);
381
bool is_expensive_processor(unsigned char *arg);
358
383
friend class Item_ref_null_helper;
359
384
friend class Item_is_not_null_test;
374
399
Item_allany_subselect(Item * left_expr, chooser_compare_func_creator fc,
375
st_select_lex *select_lex, bool all);
400
Select_Lex *select_lex, bool all);
377
402
// only ALL subquery has upper not
378
403
subs_type substype() { return all?ALL_SUBS:ANY_SUBS; }
379
trans_res select_transformer(JOIN *join);
404
trans_res select_transformer(Join *join);
380
405
virtual void print(String *str, enum_query_type query_type);
384
class subselect_engine: public Sql_alloc
409
class subselect_engine: public memory::SqlAlloc
387
412
select_result_interceptor *result; /* results storage class */
388
THD *thd; /* pointer to current THD */
413
Session *session; /* pointer to current Session */
389
414
Item_subselect *item; /* item, that use this engine */
390
415
enum Item_result res_type; /* type of results */
391
416
enum_field_types res_field_type; /* column type of the results */
405
430
res_field_type= DRIZZLE_TYPE_VARCHAR;
408
virtual ~subselect_engine() {}; // to satisfy compiler
433
virtual ~subselect_engine() {} // to satisfy compiler
409
434
virtual void cleanup()= 0;
412
Also sets "thd" for subselect_engine::result.
437
Also sets "session" for subselect_engine::result.
413
438
Should be called before prepare().
415
void set_thd(THD *thd_arg);
416
THD * get_thd() { return thd; }
440
void set_session(Session *session_arg);
441
Session * get_session() { return session; }
417
442
virtual int prepare()= 0;
418
443
virtual void fix_length_and_dec(Item_cache** row)= 0;
437
462
caller should call exec() again for the new engine.
439
464
virtual int exec()= 0;
440
virtual uint cols()= 0; /* return number of columns in select */
441
virtual uint8_t uncacheable()= 0; /* query is uncacheable */
465
virtual uint32_t cols()= 0; /* return number of columns in select */
466
virtual bool uncacheable()= 0; /* query is uncacheable */
467
virtual bool uncacheable(uint32_t bit_pos)= 0; /* query is uncacheable */
442
468
enum Item_result type() { return res_type; }
443
469
enum_field_types field_type() { return res_field_type; }
444
470
virtual void exclude()= 0;
445
virtual bool may_be_null() { return maybe_null; };
471
virtual bool may_be_null() { return maybe_null; }
446
472
virtual table_map upper_select_const_tables()= 0;
447
static table_map calc_const_tables(TABLE_LIST *);
473
static table_map calc_const_tables(TableList *);
448
474
virtual void print(String *str, enum_query_type query_type)= 0;
449
475
virtual bool change_result(Item_subselect *si,
450
476
select_result_interceptor *result)= 0;
464
490
bool prepared; /* simple subselect is prepared */
465
491
bool optimized; /* simple subselect is optimized */
466
492
bool executed; /* simple subselect is executed */
467
st_select_lex *select_lex; /* corresponding select_lex */
468
JOIN * join; /* corresponding JOIN structure */
493
Select_Lex *select_lex; /* corresponding select_lex */
494
Join * join; /* corresponding JOIN structure */
470
subselect_single_select_engine(st_select_lex *select,
496
subselect_single_select_engine(Select_Lex *select,
471
497
select_result_interceptor *result,
472
498
Item_subselect *item);
475
501
void fix_length_and_dec(Item_cache** row);
478
uint8_t uncacheable();
505
bool uncacheable(uint32_t bit_pos);
480
507
table_map upper_select_const_tables();
481
508
virtual void print (String *str, enum_query_type query_type);
494
521
class subselect_union_engine: public subselect_engine
496
st_select_lex_unit *unit; /* corresponding unit structure */
523
Select_Lex_Unit *unit; /* corresponding unit structure */
498
subselect_union_engine(st_select_lex_unit *u,
525
subselect_union_engine(Select_Lex_Unit *u,
499
526
select_result_interceptor *result,
500
527
Item_subselect *item);
503
530
void fix_length_and_dec(Item_cache** row);
506
uint8_t uncacheable();
534
bool uncacheable(uint32_t bit_pos);
508
536
table_map upper_select_const_tables();
509
537
virtual void print (String *str, enum_query_type query_type);
523
551
lookup in a unique index.
525
553
This engine is used to resolve subqueries in forms
527
outer_expr IN (SELECT tbl.unique_key FROM tbl WHERE subq_where)
555
outer_expr IN (SELECT tbl.unique_key FROM tbl WHERE subq_where)
531
559
(oe1, .. oeN) IN (SELECT uniq_key_part1, ... uniq_key_partK
532
FROM tbl WHERE subqwhere)
560
FROM tbl WHERE subqwhere)
534
562
i.e. the subquery is a single table SELECT without GROUP BY, aggregate
548
576
bool null_keypart; /* TRUE <=> constructed search tuple has a NULL */
551
// constructor can assign THD because it will be called after JOIN::prepare
552
subselect_uniquesubquery_engine(THD *thd_arg, st_join_table *tab_arg,
579
// constructor can assign Session because it will be called after Join::prepare
580
subselect_uniquesubquery_engine(Session *session_arg, JoinTable *tab_arg,
553
581
Item_subselect *subs, Item *where)
554
582
:subselect_engine(subs, 0), tab(tab_arg), cond(where)
584
set_session(session_arg);
560
588
void fix_length_and_dec(Item_cache** row);
562
uint cols() { return 1; }
563
uint8_t uncacheable() { return UNCACHEABLE_DEPENDENT; }
590
uint32_t cols() { return 1; }
591
bool uncacheable() { return true; }
592
bool uncacheable(uint32_t) { return true; }
565
594
table_map upper_select_const_tables() { return 0; }
566
595
virtual void print (String *str, enum_query_type query_type);
578
607
/* FALSE for 'ref', TRUE for 'ref-or-null'. */
581
610
The "having" clause. This clause (further reffered to as "artificial
582
having") was inserted by subquery transformation code. It contains
583
Item(s) that have a side-effect: they record whether the subquery has
611
having") was inserted by subquery transformation code. It contains
612
Item(s) that have a side-effect: they record whether the subquery has
584
613
produced a row with NULL certain components. We need to use it for cases
586
615
(oe1, oe2) IN (SELECT t.key, t.no_key FROM t1)
587
616
where we do index lookup on t.key=oe1 but need also to check if there
588
617
was a row such that t.no_key IS NULL.
590
619
NOTE: This is currently here and not in the uniquesubquery_engine. Ideally
591
620
it should have been in uniquesubquery_engine in order to allow execution of
594
623
(oe1, oe2) IN (SELECT primary_key, non_key_maybe_null_field FROM tbl)
596
625
We could use uniquesubquery_engine for the first component and let
603
632
The above example subquery is handled as a full-blown SELECT with eq_ref
604
633
access to one table.
606
Due to this limitation, the "artificial having" currently needs to be
635
Due to this limitation, the "artificial having" currently needs to be
607
636
checked by only in indexsubquery_engine.
612
// constructor can assign THD because it will be called after JOIN::prepare
613
subselect_indexsubquery_engine(THD *thd_arg, st_join_table *tab_arg,
641
// constructor can assign Session because it will be called after Join::prepare
642
subselect_indexsubquery_engine(Session *session_arg, JoinTable *tab_arg,
614
643
Item_subselect *subs, Item *where,
615
644
Item *having_arg, bool chk_null)
616
:subselect_uniquesubquery_engine(thd_arg, tab_arg, subs, where),
645
:subselect_uniquesubquery_engine(session_arg, tab_arg, subs, where),
617
646
check_null(chk_null),
618
647
having(having_arg)
657
686
QEP to execute the subquery and materialize its result into a
658
687
temporary table. Created during the first call to exec().
660
JOIN *materialize_join;
689
Join *materialize_join;
661
690
/* Temp table context of the outer select's JOIN. */
662
TMP_TABLE_PARAM *tmp_param;
691
Tmp_Table_Param *tmp_param;
665
subselect_hash_sj_engine(THD *thd, Item_subselect *in_predicate,
694
subselect_hash_sj_engine(Session *session_in, Item_subselect *in_predicate,
666
695
subselect_single_select_engine *old_engine)
667
:subselect_uniquesubquery_engine(thd, NULL, in_predicate, NULL),
696
:subselect_uniquesubquery_engine(session_in, NULL, in_predicate, NULL),
668
697
is_materialized(false), materialize_engine(old_engine),
669
698
materialize_join(NULL), tmp_param(NULL)
676
705
int prepare() { return 0; }
678
707
virtual void print (String *str, enum_query_type query_type);
681
710
return materialize_engine->cols();
683
712
virtual enum_engine_type engine_type() { return HASH_SJ_ENGINE; }
715
} /* namespace drizzled */
717
#endif /* DRIZZLED_ITEM_SUBSELECT_H */