sql_select.h 65.1 KB
Newer Older
1 2 3
#ifndef SQL_SELECT_INCLUDED
#define SQL_SELECT_INCLUDED

4 5
/* Copyright (c) 2000, 2013, Oracle and/or its affiliates.
   Copyright (c) 2008, 2013, Monty Program Ab.
unknown's avatar
unknown committed
6

unknown's avatar
unknown committed
7 8
   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
unknown's avatar
unknown committed
9
   the Free Software Foundation; version 2 of the License.
unknown's avatar
unknown committed
10

unknown's avatar
unknown committed
11 12 13 14
   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.
unknown's avatar
unknown committed
15

unknown's avatar
unknown committed
16 17
   You should have received a copy of the GNU General Public License
   along with this program; if not, write to the Free Software
Kent Boortz's avatar
Kent Boortz committed
18
   Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301  USA */
unknown's avatar
unknown committed
19

unknown's avatar
unknown committed
20 21 22 23 24 25
/**
  @file

  @brief
  classes to use when handling where clause
*/
unknown's avatar
unknown committed
26

27
#ifdef USE_PRAGMA_INTERFACE
unknown's avatar
unknown committed
28 29 30 31
#pragma interface			/* gcc class implementation */
#endif

#include "procedure.h"
32 33 34 35
#include "sql_array.h"                        /* Array */
#include "records.h"                          /* READ_RECORD */
#include "opt_range.h"                /* SQL_SELECT, QUICK_SELECT_I */

unknown's avatar
unknown committed
36

37 38 39
/* Values in optimize */
#define KEY_OPTIMIZE_EXISTS		1
#define KEY_OPTIMIZE_REF_OR_NULL	2
40 41 42 43 44
#define KEY_OPTIMIZE_EQ	                4

inline uint get_hash_join_key_no() { return MAX_KEY; }

inline bool is_hash_join_key_no(uint key) { return key == MAX_KEY; }
45

unknown's avatar
unknown committed
46 47
typedef struct keyuse_t {
  TABLE *table;
unknown's avatar
unknown committed
48
  Item	*val;				/**< or value if no field */
unknown's avatar
unknown committed
49
  table_map used_tables;
50
  uint	key, keypart, optimize;
unknown's avatar
unknown committed
51 52
  key_part_map keypart_map;
  ha_rows      ref_table_rows;
unknown's avatar
unknown committed
53
  /**
54 55 56 57
    If true, the comparison this value was created from will not be
    satisfied if val has NULL 'value'.
  */
  bool null_rejecting;
58 59 60 61 62 63 64 65 66 67 68
  /*
    !NULL - This KEYUSE was created from an equality that was wrapped into
            an Item_func_trig_cond. This means the equality (and validity of 
            this KEYUSE element) can be turned on and off. The on/off state 
            is indicted by the pointed value:
              *cond_guard == TRUE <=> equality condition is on
              *cond_guard == FALSE <=> equality condition is off

    NULL  - Otherwise (the source equality can't be turned off)
  */
  bool *cond_guard;
69 70 71 72 73
  /*
     0..64    <=> This was created from semi-join IN-equality # sj_pred_no.
     MAX_UINT  Otherwise
  */
  uint         sj_pred_no;
74 75

  bool is_for_hash_join() { return is_hash_join_key_no(key); }
unknown's avatar
unknown committed
76 77
} KEYUSE;

Igor Babaev's avatar
Igor Babaev committed
78 79
#define NO_KEYPART ((uint)(-1))

unknown's avatar
unknown committed
80 81
class store_key;

82 83
const int NO_REF_PART= uint(-1);

unknown's avatar
unknown committed
84 85 86
typedef struct st_table_ref
{
  bool		key_err;
Konstantin Osipov's avatar
Konstantin Osipov committed
87 88
  /** True if something was read into buffer in join_read_key.  */
  bool          has_record;
unknown's avatar
unknown committed
89 90 91 92 93
  uint          key_parts;                ///< num of ...
  uint          key_length;               ///< length of key_buff
  int           key;                      ///< key no
  uchar         *key_buff;                ///< value to look for with key
  uchar         *key_buff2;               ///< key_buff+key_length
unknown's avatar
unknown committed
94
  store_key     **key_copy;               //
95 96 97 98 99 100 101

  /*
    Bitmap of key parts which refer to constants. key_copy only has copiers for
    non-const key parts.
  */
  key_part_map  const_ref_part_map;

unknown's avatar
unknown committed
102
  Item          **items;                  ///< val()'s for each keypart
103 104 105 106 107 108 109 110 111 112 113 114
  /*  
    Array of pointers to trigger variables. Some/all of the pointers may be
    NULL.  The ref access can be used iff
    
      for each used key part i, (!cond_guards[i] || *cond_guards[i]) 

    This array is used by subquery code. The subquery code may inject
    triggered conditions, i.e. conditions that can be 'switched off'. A ref 
    access created from such condition is not valid when at least one of the 
    underlying conditions is switched off (see subquery code for more details)
  */
  bool          **cond_guards;
unknown's avatar
unknown committed
115
  /**
116 117 118 119
    (null_rejecting & (1<<i)) means the condition is '=' and no matching
    rows will be produced if items[i] IS NULL (see add_not_null_conds())
  */
  key_part_map  null_rejecting;
unknown's avatar
unknown committed
120
  table_map	depend_map;		  ///< Table depends on these tables.
121

122
  /* null byte position in the key_buf. Used for REF_OR_NULL optimization */
123
  uchar          *null_ref_key;
124
  /* 
125 126 127
    ref_or_null optimization: number of key part that alternates between
    the lookup value or NULL (there's only one such part). 
    If we're not using ref_or_null, the value is NO_REF_PART
128 129 130
  */
  uint           null_ref_part;

Konstantin Osipov's avatar
Konstantin Osipov committed
131 132 133 134 135
  /*
    The number of times the record associated with this key was used
    in the join.
  */
  ha_rows       use_count;
unknown's avatar
unknown committed
136

137 138 139
  /*
    TRUE <=> disable the "cache" as doing lookup with the same key value may
    produce different results (because of Index Condition Pushdown)
unknown's avatar
unknown committed
140

141 142
  */
  bool          disable_cache;
unknown's avatar
unknown committed
143

unknown's avatar
unknown committed
144
  bool tmp_table_index_lookup_init(THD *thd, KEY *tmp_key, Item_iterator &it,
unknown's avatar
unknown committed
145
                                   bool value, uint skip= 0);
146
  bool is_access_triggered();
unknown's avatar
unknown committed
147 148 149 150
} TABLE_REF;


/*
151
  The structs which holds the join connections and join states
unknown's avatar
unknown committed
152 153
*/
enum join_type { JT_UNKNOWN,JT_SYSTEM,JT_CONST,JT_EQ_REF,JT_REF,JT_MAYBE_REF,
154
		 JT_ALL, JT_RANGE, JT_NEXT, JT_FT, JT_REF_OR_NULL,
155
		 JT_UNIQUE_SUBQUERY, JT_INDEX_SUBQUERY, JT_INDEX_MERGE,
156
                 JT_HASH, JT_HASH_RANGE, JT_HASH_NEXT, JT_HASH_INDEX_MERGE};
unknown's avatar
unknown committed
157 158 159

class JOIN;

160 161 162 163 164 165 166
enum enum_nested_loop_state
{
  NESTED_LOOP_KILLED= -2, NESTED_LOOP_ERROR= -1,
  NESTED_LOOP_OK= 0, NESTED_LOOP_NO_MORE_ROWS= 1,
  NESTED_LOOP_QUERY_LIMIT= 3, NESTED_LOOP_CURSOR_LIMIT= 4
};

167

168 169 170 171 172 173 174 175 176 177 178
/* Possible sj_strategy values */
enum sj_strategy_enum
{
  SJ_OPT_NONE=0,
  SJ_OPT_DUPS_WEEDOUT=1,
  SJ_OPT_LOOSE_SCAN  =2,
  SJ_OPT_FIRST_MATCH =3,
  SJ_OPT_MATERIALIZE =4,
  SJ_OPT_MATERIALIZE_SCAN=5
};

179 180 181 182 183 184
/* Values for JOIN_TAB::packed_info */
#define TAB_INFO_HAVE_VALUE 1
#define TAB_INFO_USING_INDEX 2
#define TAB_INFO_USING_WHERE 4
#define TAB_INFO_FULL_SCAN_ON_NULL 8

185 186
typedef enum_nested_loop_state
(*Next_select_func)(JOIN *, struct st_join_table *, bool);
187
Next_select_func setup_end_select_func(JOIN *join);
188
int rr_sequential(READ_RECORD *info);
189
int rr_sequential_and_unpack(READ_RECORD *info);
190

191

192
#include "sql_explain.h"
Sergey Petrunya's avatar
Sergey Petrunya committed
193 194 195 196 197

/**************************************************************************************
 * New EXPLAIN structures END
 *************************************************************************************/

198
class JOIN_CACHE;
199
class SJ_TMP_TABLE;
200
class JOIN_TAB_RANGE;
201

unknown's avatar
unknown committed
202
typedef struct st_join_table {
203
  st_join_table() {}                          /* Remove gcc warning */
unknown's avatar
unknown committed
204
  TABLE		*table;
unknown's avatar
unknown committed
205
  KEYUSE	*keyuse;			/**< pointer to first used key */
206 207
  KEY           *hj_key;       /**< descriptor of the used best hash join key
				    not supported by any index                 */
unknown's avatar
unknown committed
208 209
  SQL_SELECT	*select;
  COND		*select_cond;
210 211
  COND          *on_precond;    /**< part of on condition to check before
				     accessing the first inner table           */  
unknown's avatar
unknown committed
212
  QUICK_SELECT_I *quick;
213 214 215 216 217 218 219 220
  /* 
    The value of select_cond before we've attempted to do Index Condition
    Pushdown. We may need to restore everything back if we first choose one
    index but then reconsider (see test_if_skip_sort_order() for such
    scenarios).
    NULL means no index condition pushdown was performed.
  */
  Item          *pre_idx_push_select_cond;
221 222 223 224 225 226 227
  /*
    Pointer to the associated ON expression. on_expr_ref=!NULL except for
    degenerate joins. 
    *on_expr_ref!=NULL for tables that are first inner tables within an outer
    join.
  */
  Item	       **on_expr_ref;
unknown's avatar
unknown committed
228 229 230 231 232 233 234
  COND_EQUAL    *cond_equal;    /**< multiple equalities for the on expression */
  st_join_table *first_inner;   /**< first inner table for including outerjoin */
  bool           found;         /**< true after all matches or null complement */
  bool           not_null_compl;/**< true before null complement is added      */
  st_join_table *last_inner;    /**< last table table for embedding outer join */
  st_join_table *first_upper;  /**< first inner table for embedding outer join */
  st_join_table *first_unmatched; /**< used for optimization purposes only     */
235 236

  /*
Sergey Petrunya's avatar
Sergey Petrunya committed
237
    For join tabs that are inside an SJM bush: root of the bush
238 239 240
  */
  st_join_table *bush_root_tab;

Sergey Petrunya's avatar
Sergey Petrunya committed
241
  /* TRUE <=> This join_tab is inside an SJM bush and is the last leaf tab here */
242 243 244 245 246 247 248
  bool          last_leaf_in_bush;
  
  /*
    ptr  - this is a bush, and ptr points to description of child join_tab
           range
    NULL - this join tab has no bush children
  */
249
  JOIN_TAB_RANGE *bush_children;
250 251
  
  /* Special content for EXPLAIN 'Extra' column or NULL if none */
252
  enum explain_extra_tag info;
Sergei Petrunia's avatar
Sergei Petrunia committed
253
  
Sergei Petrunia's avatar
Sergei Petrunia committed
254
  Table_access_tracker *tracker;
255
  Table_access_tracker *jbuf_tracker;
256 257 258 259 260 261
  /* 
    Bitmap of TAB_INFO_* bits that encodes special line for EXPLAIN 'Extra'
    column, or 0 if there is no info.
  */
  uint          packed_info;

Konstantin Osipov's avatar
Konstantin Osipov committed
262
  READ_RECORD::Setup_func read_first_record;
263
  Next_select_func next_select;
unknown's avatar
unknown committed
264
  READ_RECORD	read_record;
265 266 267 268 269
  /* 
    Currently the following two fields are used only for a [NOT] IN subquery
    if it is executed by an alternative full table scan when the left operand of
    the subquery predicate is evaluated to NULL.
  */  
Konstantin Osipov's avatar
Konstantin Osipov committed
270 271
  READ_RECORD::Setup_func save_read_first_record;/* to save read_first_record */
  READ_RECORD::Read_func save_read_record;/* to save read_record.read_record */
unknown's avatar
unknown committed
272
  double	worst_seeks;
unknown's avatar
unknown committed
273 274
  key_map	const_keys;			/**< Keys with constant part */
  key_map	checked_keys;			/**< Keys checked in find_best */
unknown's avatar
unknown committed
275
  key_map	needed_reg;
unknown's avatar
unknown committed
276
  key_map       keys;                           /**< all keys with can be used */
277 278 279 280 281 282 283 284 285 286 287 288 289

  /* Either #rows in the table or 1 for const table.  */
  ha_rows	records;
  /*
    Number of records that will be scanned (yes scanned, not returned) by the
    best 'independent' access method, i.e. table scan or QUICK_*_SELECT)
  */
  ha_rows       found_records;
  /*
    Cost of accessing the table using "ALL" or range/index_merge access
    method (but not 'index' for some reason), i.e. this matches method which
    E(#records) is in found_records.
  */
290
  double        read_time;
291
  
292
  /* Copy of POSITION::records_read, set by get_best_combination() */
Igor Babaev's avatar
Igor Babaev committed
293
  double        records_read;
294
  
Igor Babaev's avatar
Igor Babaev committed
295 296
  /* The selectivity of the conditions that can be pushed to the table */ 
  double        cond_selectivity;  
Igor Babaev's avatar
Igor Babaev committed
297
  
298 299
  /* Startup cost for execution */
  double        startup_cost;
300 301 302
    
  double        partial_join_cardinality;

unknown's avatar
unknown committed
303
  table_map	dependent,key_dependent;
304 305 306 307 308 309 310 311 312 313
  /*
     1 - use quick select
     2 - use "Range checked for each record"
  */
  uint		use_quick;
  /*
    Index to use. Note: this is valid only for 'index' access, but not range or
    ref access.
  */
  uint          index;
unknown's avatar
unknown committed
314
  uint		status;				///< Save status for cache
315 316 317 318
  uint		used_fields;
  ulong         used_fieldlength;
  ulong         max_used_fieldlength;
  uint          used_blobs;
319 320
  uint          used_null_fields;
  uint          used_uneven_bit_fields;
unknown's avatar
unknown committed
321
  enum join_type type;
Igor Babaev's avatar
Igor Babaev committed
322 323 324
  bool          cached_eq_ref_table,eq_ref_table;
  bool          shortcut_for_distinct;
  bool          sorted;
unknown's avatar
unknown committed
325 326 327 328 329 330
  /* 
    If it's not 0 the number stored this field indicates that the index
    scan has been chosen to access the table data and we expect to scan 
    this number of rows for the table.
  */ 
  ha_rows       limit; 
unknown's avatar
unknown committed
331
  TABLE_REF	ref;
332 333 334 335 336 337 338
  /* TRUE <=> condition pushdown supports other tables presence */
  bool          icp_other_tables_ok;
  /* 
    TRUE <=> condition pushed to the index has to be factored out of
    the condition pushed to the table
  */
  bool          idx_cond_fact_out;
339
  bool          use_join_cache;
340
  uint          used_join_cache_level;
341
  ulong         join_buffer_size_limit;
342 343 344 345 346 347
  JOIN_CACHE	*cache;
  /*
    Index condition for BKA access join
  */
  Item          *cache_idx_cond;
  SQL_SELECT    *cache_select;
348
  JOIN		*join;
349 350 351 352 353
  /*
    Embedding SJ-nest (may be not the direct parent), or NULL if none.
    This variable holds the result of table pullout.
  */
  TABLE_LIST    *emb_sj_nest;
Sergey Petrunya's avatar
Sergey Petrunya committed
354

355 356 357 358
  /* FirstMatch variables (final QEP) */
  struct st_join_table *first_sj_inner_tab;
  struct st_join_table *last_sj_inner_tab;

359 360 361
  /* Variables for semi-join duplicate elimination */
  SJ_TMP_TABLE  *flush_weedout_table;
  SJ_TMP_TABLE  *check_weed_out_table;
362 363
  /* for EXPLAIN only: */
  SJ_TMP_TABLE  *first_weedout_table;
364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380
  
  /*
    If set, means we should stop join enumeration after we've got the first
    match and return to the specified join tab. May point to
    join->join_tab[-1] which means stop join execution after the first
    match.
  */
  struct st_join_table  *do_firstmatch;
 
  /* 
     ptr  - We're doing a LooseScan, this join tab is the first (i.e. 
            "driving") join tab), and ptr points to the last join tab
            handled by the strategy. loosescan_match_tab->found_match
            should be checked to see if the current value group had a match.
     NULL - Not doing a loose scan on this join tab.
  */
  struct st_join_table *loosescan_match_tab;
381 382 383
  
  /* TRUE <=> we are inside LooseScan range */
  bool inside_loosescan_range;
384 385 386 387

  /* Buffer to save index tuple to be able to skip duplicates */
  uchar *loosescan_buf;
  
388 389 390 391 392 393
  /* 
    Index used by LooseScan (we store it here separately because ref access
    stores it in tab->ref.key, while range scan stores it in tab->index, etc)
  */
  uint loosescan_key;

394 395 396 397 398 399 400 401 402 403 404 405 406
  /* Length of key tuple (depends on #keyparts used) to store in the above */
  uint loosescan_key_len;

  /* Used by LooseScan. TRUE<=> there has been a matching record combination */
  bool found_match;
  
  /*
    Used by DuplicateElimination. tab->table->ref must have the rowid
    whenever we have a current record.
  */
  int  keep_current_rowid;

  /* NestedOuterJoins: Bitmap of nested joins this table is part of */
407
  nested_join_map embedding_map;
unknown's avatar
unknown committed
408

409 410 411
  /*
    Semi-join strategy to be used for this join table. This is a copy of
    POSITION::sj_strategy field. This field is set up by the
412
    fix_semijoin_strategies_for_picked_join_order.
413
  */
414
  enum sj_strategy_enum sj_strategy;
415

Sergey Petrunya's avatar
Sergey Petrunya committed
416
  uint n_sj_tables;
417

418 419
  bool preread_init_done;

420
  void cleanup();
421 422 423 424 425 426
  inline bool is_using_loose_index_scan()
  {
    return (select && select->quick &&
            (select->quick->get_type() ==
             QUICK_SELECT_I::QS_TYPE_GROUP_MIN_MAX));
  }
427 428 429 430 431
  bool is_using_agg_loose_index_scan ()
  {
    return (is_using_loose_index_scan() &&
            ((QUICK_GROUP_MIN_MAX_SELECT *)select->quick)->is_agg_distinct());
  }
432 433 434 435
  bool is_inner_table_of_semi_join_with_first_match()
  {
    return first_sj_inner_tab != NULL;
  }
Igor Babaev's avatar
Igor Babaev committed
436 437 438 439
  bool is_inner_table_of_semijoin()
  {
    return emb_sj_nest != NULL;
  }
440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461
  bool is_inner_table_of_outer_join()
  {
    return first_inner != NULL;
  }
  bool is_single_inner_of_semi_join_with_first_match()
  {
    return first_sj_inner_tab == this && last_sj_inner_tab == this;            
  }
  bool is_single_inner_of_outer_join()
  {
    return first_inner == this && first_inner->last_inner == this;
  }
  bool is_first_inner_for_outer_join()
  {
    return first_inner && first_inner == this;
  }
  bool use_match_flag()
  {
    return is_first_inner_for_outer_join() || first_sj_inner_tab == this ; 
  }
  bool check_only_first_match()
  {
462 463
    return is_inner_table_of_semi_join_with_first_match() ||
           (is_inner_table_of_outer_join() &&
464 465 466 467 468 469 470
            table->reginfo.not_exists_optimize);
  }
  bool is_last_inner_table()
  {
    return (first_inner && first_inner->last_inner == this) ||
           last_sj_inner_tab == this;
  }
Igor Babaev's avatar
Igor Babaev committed
471 472 473 474 475 476 477 478 479 480 481 482 483
  /*
    Check whether the table belongs to a nest of inner tables of an
    outer join or to a nest of inner tables of a semi-join
  */
  bool is_nested_inner()
  {
    if (first_inner && 
        (first_inner != first_inner->last_inner || first_inner->first_upper))
      return TRUE;
    if (first_sj_inner_tab && first_sj_inner_tab != last_sj_inner_tab)
      return TRUE;
    return FALSE;
  }
484 485 486 487 488 489
  struct st_join_table *get_first_inner_table()
  {
    if (first_inner)
      return first_inner;
    return first_sj_inner_tab; 
  }
490 491 492 493 494 495 496 497 498 499 500 501 502 503
  void set_select_cond(COND *to, uint line)
  {
    DBUG_PRINT("info", ("select_cond changes %p -> %p at line %u tab %p",
                        select_cond, to, line, this));
    select_cond= to;
  }
  COND *set_cond(COND *new_cond)
  {
    COND *tmp_select_cond= select_cond;
    set_select_cond(new_cond, __LINE__);
    if (select)
      select->cond= new_cond;
    return tmp_select_cond;
  }
504 505
  void calc_used_field_length(bool max_fl);
  ulong get_used_fieldlength()
506
  {
507 508 509
    if (!used_fieldlength)
      calc_used_field_length(FALSE);
    return used_fieldlength;
510
  }
511
  ulong get_max_used_fieldlength()
512
  {
513 514 515
    if (!max_used_fieldlength)
      calc_used_field_length(TRUE);
    return max_used_fieldlength;
516
  }
517
  double get_partial_join_cardinality() { return partial_join_cardinality; }
Igor Babaev's avatar
Igor Babaev committed
518
  bool hash_join_is_possible();
519
  int make_scan_filter();
520 521
  bool is_ref_for_hash_join() { return is_hash_join_key_no(ref.key); }
  KEY *get_keyinfo_by_key_no(uint key) 
522
  {
523
    return (is_hash_join_key_no(key) ? hj_key : table->key_info+key);
524
  }
525
  double scan_time();
unknown's avatar
unknown committed
526
  ha_rows get_examined_rows();
527
  bool preread_init();
528

529
  bool is_sjm_nest() { return MY_TEST(bush_children); }
Igor Babaev's avatar
Igor Babaev committed
530 531 532 533 534 535 536 537 538 539

  bool access_from_tables_is_allowed(table_map used_tables,
                                     table_map sjm_lookup_tables)
  {
    table_map used_sjm_lookup_tables= used_tables & sjm_lookup_tables;
    return !used_sjm_lookup_tables ||
           (emb_sj_nest && 
            !(used_sjm_lookup_tables & ~emb_sj_nest->sj_inner_tables));
  }

540
  void remove_redundant_bnl_scan_conds();
541 542 543 544 545

  void save_explain_data(Explain_table_access *eta, table_map prefix_tables, 
                         bool distinct, struct st_join_table *first_top_tab);

  void update_explain_data(uint idx);
unknown's avatar
unknown committed
546
} JOIN_TAB;
547 548


549
#include "sql_join_cache.h"
550

551 552 553 554
enum_nested_loop_state sub_select_cache(JOIN *join, JOIN_TAB *join_tab, bool
                                        end_of_records);
enum_nested_loop_state sub_select(JOIN *join,JOIN_TAB *join_tab, bool
                                  end_of_records);
555 556 557 558 559 560 561
enum_nested_loop_state
end_send_group(JOIN *join, JOIN_TAB *join_tab __attribute__((unused)),
	       bool end_of_records);
enum_nested_loop_state
end_write_group(JOIN *join, JOIN_TAB *join_tab __attribute__((unused)),
		bool end_of_records);

562

563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767
struct st_position;

class Semi_join_strategy_picker
{
public:
  /* Called when starting to build a new join prefix */
  virtual void set_empty() = 0;

  /* 
    Update internal state after another table has been added to the join
    prefix
  */
  virtual void set_from_prev(struct st_position *prev) = 0;
  
  virtual bool check_qep(JOIN *join,
                         uint idx,
                         table_map remaining_tables, 
                         const JOIN_TAB *new_join_tab,
                         double *record_count,
                         double *read_time,
                         table_map *handled_fanout,
                         sj_strategy_enum *strategy,
                         struct st_position *loose_scan_pos) = 0;

  virtual void mark_used() = 0;

  virtual ~Semi_join_strategy_picker() {} 
};


/*
  Duplicate Weedout strategy optimization state
*/

class Duplicate_weedout_picker : public Semi_join_strategy_picker
{
  /* The first table that the strategy will need to handle */
  uint  first_dupsweedout_table;

  /*
    Tables that we will need to have in the prefix to do the weedout step
    (all inner and all outer that the involved semi-joins are correlated with)
  */
  table_map dupsweedout_tables;
  
  bool is_used;
public:
  void set_empty()
  {
    dupsweedout_tables= 0;
    first_dupsweedout_table= MAX_TABLES;
    is_used= FALSE;
  }
  void set_from_prev(struct st_position *prev);
  
  bool check_qep(JOIN *join,
                 uint idx,
                 table_map remaining_tables, 
                 const JOIN_TAB *new_join_tab,
                 double *record_count,
                 double *read_time,
                 table_map *handled_fanout,
                 sj_strategy_enum *stratey,
                 struct st_position *loose_scan_pos);

  void mark_used() { is_used= TRUE; }
  friend void fix_semijoin_strategies_for_picked_join_order(JOIN *join);
};


class Firstmatch_picker : public Semi_join_strategy_picker
{
  /*
    Index of the first inner table that we intend to handle with this
    strategy
  */
  uint first_firstmatch_table;
  /*
    Tables that were not in the join prefix when we've started considering 
    FirstMatch strategy.
  */
  table_map first_firstmatch_rtbl;
  /* 
    Tables that need to be in the prefix before we can calculate the cost
    of using FirstMatch strategy.
   */
  table_map firstmatch_need_tables;

  bool is_used;

  bool in_firstmatch_prefix() { return (first_firstmatch_table != MAX_TABLES); }
  void invalidate_firstmatch_prefix() { first_firstmatch_table= MAX_TABLES; }
public:
  void set_empty()
  {
    invalidate_firstmatch_prefix();
    is_used= FALSE;
  }

  void set_from_prev(struct st_position *prev);
  bool check_qep(JOIN *join,
                 uint idx,
                 table_map remaining_tables, 
                 const JOIN_TAB *new_join_tab,
                 double *record_count,
                 double *read_time,
                 table_map *handled_fanout,
                 sj_strategy_enum *strategy,
                 struct st_position *loose_scan_pos);

  void mark_used() { is_used= TRUE; }
  friend void fix_semijoin_strategies_for_picked_join_order(JOIN *join);
};


class LooseScan_picker : public Semi_join_strategy_picker
{
  /* The first (i.e. driving) table we're doing loose scan for */
  uint        first_loosescan_table;
  /* 
     Tables that need to be in the prefix before we can calculate the cost
     of using LooseScan strategy.
  */
  table_map   loosescan_need_tables;

  /*
    keyno  -  Planning to do LooseScan on this key. If keyuse is NULL then 
              this is a full index scan, otherwise this is a ref+loosescan
              scan (and keyno matches the KEUSE's)
    MAX_KEY - Not doing a LooseScan
  */
  uint loosescan_key;  // final (one for strategy instance )
  uint loosescan_parts; /* Number of keyparts to be kept distinct */
  
  bool is_used;
public:
  void set_empty()
  {
    first_loosescan_table= MAX_TABLES; 
    is_used= FALSE;
  }

  void set_from_prev(struct st_position *prev);
  bool check_qep(JOIN *join,
                 uint idx,
                 table_map remaining_tables, 
                 const JOIN_TAB *new_join_tab,
                 double *record_count,
                 double *read_time,
                 table_map *handled_fanout,
                 sj_strategy_enum *strategy,
                 struct st_position *loose_scan_pos);
  void mark_used() { is_used= TRUE; }

  friend class Loose_scan_opt;
  friend void best_access_path(JOIN      *join,
                               JOIN_TAB  *s,
                               table_map remaining_tables,
                               uint      idx,
                               bool      disable_jbuf,
                               double    record_count,
                               struct st_position *pos,
                               struct st_position *loose_scan_pos);
  friend bool get_best_combination(JOIN *join);
  friend int setup_semijoin_dups_elimination(JOIN *join, ulonglong options,
                                             uint no_jbuf_after);
  friend void fix_semijoin_strategies_for_picked_join_order(JOIN *join);
};


class Sj_materialization_picker : public Semi_join_strategy_picker
{
  bool is_used;

  /* The last inner table (valid once we're after it) */
  uint      sjm_scan_last_inner;
  /*
    Tables that we need to have in the prefix to calculate the correct cost.
    Basically, we need all inner tables and outer tables mentioned in the
    semi-join's ON expression so we can correctly account for fanout.
  */
  table_map sjm_scan_need_tables;

public:
  void set_empty()
  {
    sjm_scan_need_tables= 0;
    LINT_INIT(sjm_scan_last_inner);
    is_used= FALSE;
  }
  void set_from_prev(struct st_position *prev);
  bool check_qep(JOIN *join,
                 uint idx,
                 table_map remaining_tables, 
                 const JOIN_TAB *new_join_tab,
                 double *record_count,
                 double *read_time,
                 table_map *handled_fanout,
                 sj_strategy_enum *strategy,
                 struct st_position *loose_scan_pos);
  void mark_used() { is_used= TRUE; }

  friend void fix_semijoin_strategies_for_picked_join_order(JOIN *join);
};

768

unknown's avatar
unknown committed
769
/**
770 771 772
  Information about a position of table within a join order. Used in join
  optimization.
*/
773
typedef struct st_position :public Sql_alloc
774
{
775 776 777
  /* The table that's put into join order */
  JOIN_TAB *table;

778 779 780 781 782
  /*
    The "fanout": number of output rows that will be produced (after
    pushed down selection condition is applied) per each row combination of
    previous tables.
  */
unknown's avatar
unknown committed
783
  double records_read;
784

Igor Babaev's avatar
Igor Babaev committed
785 786
  /* The selectivity of the pushed down conditions */
  double cond_selectivity; 
Igor Babaev's avatar
Igor Babaev committed
787

788 789 790 791 792
  /* 
    Cost accessing the table in course of the entire complete join execution,
    i.e. cost of one access method use (e.g. 'range' or 'ref' scan ) times 
    number the access method will be invoked.
  */
793
  double read_time;
794 795

  /* Cumulative cost and record count for the join prefix */
796
  Cost_estimate prefix_cost;
797
  double    prefix_record_count;
798 799 800 801 802

  /*
    NULL  -  'index' or 'range' or 'index_merge' or 'ALL' access is used.
    Other - [eq_]ref[_or_null] access is used. Pointer to {t.keypart1 = expr}
  */
unknown's avatar
unknown committed
803
  KEYUSE *key;
804 805 806

  /* If ref-based access is used: bitmap of tables this table depends on  */
  table_map ref_depend_map;
807 808 809 810 811 812 813
 
  /*
    TRUE <=> join buffering will be used. At the moment this is based on 
    *very* imprecise guesses made in best_access_path(). 
  */
  bool use_join_buffer;
 
814 815 816 817 818 819 820 821 822 823 824 825
  /*
    Current optimization state: Semi-join strategy to be used for this
    and preceding join tables.
    
    Join optimizer sets this for the *last* join_tab in the
    duplicate-generating range. That is, in order to interpret this field, 
    one needs to traverse join->[best_]positions array from right to left.
    When you see a join table with sj_strategy!= SJ_OPT_NONE, some other
    field (depending on the strategy) tells how many preceding positions 
    this applies to. The values of covered_preceding_positions->sj_strategy
    must be ignored.
  */
826 827
  enum sj_strategy_enum sj_strategy;
  
828 829 830 831 832 833 834
  /*
    Valid only after fix_semijoin_strategies_for_picked_join_order() call:
    if sj_strategy!=SJ_OPT_NONE, this is the number of subsequent tables that
    are covered by the specified semi-join strategy
  */
  uint n_sj_tables;

835
  table_map prefix_dups_producing_tables;
unknown's avatar
unknown committed
836

837 838 839 840 841 842 843
  table_map inner_tables_handled_with_other_sjs;
   
  Duplicate_weedout_picker  dups_weedout_picker;
  Firstmatch_picker         firstmatch_picker;
  LooseScan_picker          loosescan_picker;
  Sj_materialization_picker sjmat_picker;
} POSITION;
844

845 846 847 848
typedef struct st_rollup
{
  enum State { STATE_NONE, STATE_INITED, STATE_READY };
  State state;
unknown's avatar
unknown committed
849
  Item_null_result **null_items;
850 851 852 853
  Item ***ref_pointer_arrays;
  List<Item> *fields;
} ROLLUP;

unknown's avatar
unknown committed
854

855 856 857 858 859 860 861
class JOIN_TAB_RANGE: public Sql_alloc
{
public:
  JOIN_TAB *start;
  JOIN_TAB *end;
};

unknown's avatar
unknown committed
862

863 864
class JOIN :public Sql_alloc
{
865
private:
unknown's avatar
unknown committed
866 867
  JOIN(const JOIN &rhs);                        /**< not implemented */
  JOIN& operator=(const JOIN &rhs);             /**< not implemented */
868 869

protected:
unknown's avatar
MWL#89  
unknown committed
870 871

  /**
unknown's avatar
MWL#89  
unknown committed
872
    The subset of the state of a JOIN that represents an optimized query
873 874
    execution plan. Allows saving/restoring different JOIN plans for the same
    query.
unknown's avatar
MWL#89  
unknown committed
875
  */
876
  class Join_plan_state {
unknown's avatar
MWL#89  
unknown committed
877
  public:
878 879
    DYNAMIC_ARRAY keyuse;        /* Copy of the JOIN::keyuse array. */
    POSITION *best_positions;    /* Copy of JOIN::best_positions */
unknown's avatar
MWL#89  
unknown committed
880
    /* Copies of the JOIN_TAB::keyuse pointers for each JOIN_TAB. */
881
    KEYUSE **join_tab_keyuse;
unknown's avatar
MWL#89  
unknown committed
882
    /* Copies of JOIN_TAB::checked_keys for each JOIN_TAB. */
883 884 885
    key_map *join_tab_checked_keys;
    SJ_MATERIALIZATION_INFO **sj_mat_info;
    my_bool error;
unknown's avatar
MWL#89  
unknown committed
886
  public:
887
    Join_plan_state(uint tables) : error(0)
unknown's avatar
MWL#89  
unknown committed
888 889 890
    {   
      keyuse.elements= 0;
      keyuse.buffer= NULL;
891 892 893 894 895 896 897 898 899 900 901
      best_positions= 0;                        /* To detect errors */
      error= my_multi_malloc(MYF(MY_WME),
                             &best_positions,
                             sizeof(*best_positions) * (tables + 1),
                             &join_tab_keyuse,
                             sizeof(*join_tab_keyuse) * tables,
                             &join_tab_checked_keys,
                             sizeof(*join_tab_checked_keys) * tables,
                             &sj_mat_info,
                             sizeof(sj_mat_info) * tables,
                             NullS) == 0;
unknown's avatar
MWL#89  
unknown committed
902
    }
903 904
    Join_plan_state(JOIN *join);
    ~Join_plan_state()
unknown's avatar
MWL#89  
unknown committed
905 906
    {
      delete_dynamic(&keyuse);
unknown's avatar
unknown committed
907
      my_free(best_positions);
unknown's avatar
MWL#89  
unknown committed
908 909 910
    }
  };

911 912 913 914 915 916 917 918
  /* Results of reoptimizing a JOIN via JOIN::reoptimize(). */
  enum enum_reopt_result {
    REOPT_NEW_PLAN, /* there is a new reoptimized plan */
    REOPT_OLD_PLAN, /* no new improved plan can be found, use the old one */
    REOPT_ERROR,    /* an irrecovarable error occured during reoptimization */
    REOPT_NONE      /* not yet reoptimized */
  };

919
  /* Support for plan reoptimization with rewritten conditions. */
unknown's avatar
MWL#89  
unknown committed
920
  enum_reopt_result reoptimize(Item *added_where, table_map join_tables,
921 922
                               Join_plan_state *save_to);
  void save_query_plan(Join_plan_state *save_to);
unknown's avatar
unknown committed
923
  void reset_query_plan();
924
  void restore_query_plan(Join_plan_state *restore_from);
925 926
  /* Choose a subquery plan for a table-less subquery. */
  bool choose_tableless_subquery_plan();
927

unknown's avatar
unknown committed
928
public:
929
  JOIN_TAB *join_tab, **best_ref;
930 931 932 933 934 935 936 937
  
  /* 
    Saved join_tab for pre_sorting. create_sort_index() will save here.. 
  */
  JOIN_TAB *pre_sort_join_tab;
  uint pre_sort_index;
  Item *pre_sort_idx_pushed_cond;
  void clean_pre_sort_join_tab();
938 939 940 941 942 943 944 945 946 947 948 949

  /*
    For "Using temporary+Using filesort" queries, JOIN::join_tab can point to
    either: 
    1. array of join tabs describing how to run the select, or
    2. array of single join tab describing read from the temporary table.

    SHOW EXPLAIN code needs to read/show #1. This is why two next members are
    there for saving it.
  */
  JOIN_TAB *table_access_tabs;
  uint     top_table_access_tabs_count;
950
  
unknown's avatar
unknown committed
951 952
  JOIN_TAB **map2table;    ///< mapping between table indexes and JOIN_TABs
  JOIN_TAB *join_tab_save; ///< saved join_tab for subquery reexecution
953 954 955

  List<JOIN_TAB_RANGE> join_tab_ranges;
  
956 957
  /*
    Base tables participating in the join. After join optimization is done, the
958 959
    tables are stored in the join order (but the only really important part is 
    that const tables are first).
960
  */
961 962 963 964 965 966 967
  TABLE    **table;
  /**
    The table which has an index that allows to produce the requried ordering.
    A special value of 0x1 means that the ordering will be produced by
    passing 1st non-const table to filesort(). NULL means no such table exists.
  */
  TABLE    *sort_by_table;
968 969 970 971 972 973 974
  /* 
    Number of tables in the join. 
    (In MySQL, it is named 'tables' and is also the number of elements in 
     join->join_tab array. In MariaDB, the latter is not true, so we've renamed
     the variable)
  */
  uint	   table_count;
975 976
  uint     outer_tables;  /**< Number of tables that are not inside semijoin */
  uint     const_tables;
977 978 979 980 981 982
  /* 
    Number of tables in the top join_tab array. Normally this matches
    (join_tab_ranges.head()->end - join_tab_ranges.head()->start). 
    
    We keep it here so that it is saved/restored with JOIN::restore_tmp.
  */
983
  uint     top_join_tab_count;
unknown's avatar
unknown committed
984
  uint	   send_group_parts;
Sergei Petrunia's avatar
Sergei Petrunia committed
985 986 987 988 989 990 991
  /*
    True if the query has GROUP BY.
    (that is, if group_by != NULL. when DISTINCT is converted into GROUP BY, it
     will set this, too. It is not clear why we need a separate var from 
     group_list)
  */
  bool	   group;
Sergei Golubchik's avatar
Sergei Golubchik committed
992 993
  bool     need_distinct;

994 995 996 997 998 999 1000
  /**
    Indicates that grouping will be performed on the result set during
    query execution. This field belongs to query execution.

    @see make_group_fields, alloc_group_fields, JOIN::exec
  */
  bool     sort_and_group; 
1001
  bool     first_record,full_join, no_field_update;
Igor Babaev's avatar
Igor Babaev committed
1002
  bool     hash_join;
1003
  bool	   do_send_rows;
Sergey Petrunya's avatar
Sergey Petrunya committed
1004
  table_map const_table_map;
Igor Babaev's avatar
Igor Babaev committed
1005 1006 1007 1008 1009
  /** 
    Bitmap of semijoin tables that the current partial plan decided
    to materialize and access by lookups
  */
  table_map sjm_lookup_tables;
Sergey Petrunya's avatar
Sergey Petrunya committed
1010 1011 1012 1013 1014
  /*
    Constant tables for which we have found a row (as opposed to those for
    which we didn't).
  */
  table_map found_const_table_map;
Sergey Petrunia's avatar
Sergey Petrunia committed
1015 1016
  
  /* Tables removed by table elimination. Set to 0 before the elimination. */
Sergey Petrunia's avatar
Sergey Petrunia committed
1017
  table_map eliminated_tables;
1018
  /*
1019 1020
     Bitmap of all inner tables from outer joins (set at start of
     make_join_statistics)
1021 1022
  */
  table_map outer_join;
Igor Babaev's avatar
Igor Babaev committed
1023 1024
  /* Bitmap of tables used in the select list items */
  table_map select_list_used_tables;
unknown's avatar
unknown committed
1025
  ha_rows  send_records,found_records,examined_rows,row_limit, select_limit;
unknown's avatar
unknown committed
1026
  /**
unknown's avatar
unknown committed
1027
    Used to fetch no more than given amount of rows per one
1028 1029 1030 1031 1032 1033 1034 1035
    fetch operation of server side cursor.
    The value is checked in end_send and end_send_group in fashion, similar
    to offset_limit_cnt:
      - fetch_limit= HA_POS_ERROR if there is no cursor.
      - when we open a cursor, we set fetch_limit to 0,
      - on each fetch iteration we add num_rows to fetch to fetch_limit
  */
  ha_rows  fetch_limit;
1036
  /* Finally picked QEP. This is result of join optimization */
1037
  POSITION *best_positions;
1038 1039 1040 1041 1042 1043 1044

/******* Join optimization state members start *******/
  /*
    pointer - we're doing optimization for a semi-join materialization nest.
    NULL    - otherwise
  */
  TABLE_LIST *emb_sjm_nest;
1045
  
1046
  /* Current join optimization state */
1047
  POSITION *positions;
1048 1049
  
  /*
1050 1051 1052 1053
    Bitmap of nested joins embedding the position at the end of the current 
    partial join (valid only during join optimizer run).
  */
  nested_join_map cur_embedding_map;
1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067
  
  /*
    Bitmap of inner tables of semi-join nests that have a proper subset of
    their tables in the current join prefix. That is, of those semi-join
    nests that have their tables both in and outside of the join prefix.
  */
  table_map cur_sj_inner_tables;
  
  /*
    Bitmap of semi-join inner tables that are in the join prefix and for
    which there's no provision for how to eliminate semi-join duplicates
    they produce.
  */
  table_map cur_dups_producing_tables;
1068
  
1069 1070
  /* We also maintain a stack of join optimization states in * join->positions[] */
/******* Join optimization state members end *******/
1071 1072 1073 1074 1075 1076 1077

  /*
    Tables within complex firstmatch ranges (i.e. those where inner tables are
    interleaved with outer tables). Join buffering cannot be used for these.
  */
  table_map complex_firstmatch_tables;

1078 1079 1080 1081 1082
  /*
    The cost of best complete join plan found so far during optimization,
    after optimization phase - cost of picked join order (not taking into
    account the changes made by test_if_skip_sort_order()).
  */
unknown's avatar
unknown committed
1083
  double   best_read;
unknown's avatar
MWL#89  
unknown committed
1084
  /*
1085
    Estimated result rows (fanout) of the join operation. If this is a subquery
unknown's avatar
MWL#89  
unknown committed
1086 1087 1088 1089 1090
    that is reexecuted multiple times, this value includes the estiamted # of
    reexecutions. This value is equal to the multiplication of all
    join->positions[i].records_read of a JOIN.
  */
  double   record_count;
unknown's avatar
unknown committed
1091
  List<Item> *fields;
unknown's avatar
unknown committed
1092
  List<Cached_item> group_fields, group_fields_cache;
unknown's avatar
unknown committed
1093
  TABLE    *tmp_table;
unknown's avatar
unknown committed
1094
  /// used to store 2 possible tmp table of SELECT
1095
  TABLE    *exec_tmp_table1, *exec_tmp_table2;
unknown's avatar
unknown committed
1096
  THD	   *thd;
1097
  Item_sum  **sum_funcs, ***sum_funcs_end;
unknown's avatar
unknown committed
1098
  /** second copy of sumfuncs (for queries with 2 temporary tables */
1099
  Item_sum  **sum_funcs2, ***sum_funcs_end2;
unknown's avatar
unknown committed
1100 1101
  Procedure *procedure;
  Item	    *having;
unknown's avatar
unknown committed
1102 1103
  Item      *tmp_having; ///< To store having when processed temporary table
  Item      *having_history; ///< Store having for explain
1104
  ulonglong  select_options;
1105 1106 1107 1108 1109 1110 1111 1112 1113
  /* 
    Bitmap of allowed types of the join caches that
    can be used for join operations
  */
  uint allowed_join_cache_types;
  bool allowed_semijoin_with_cache;
  bool allowed_outer_join_with_cache;
  /* Maximum level of the join caches that can be used for join operations */ 
  uint max_allowed_join_cache_level;
unknown's avatar
unknown committed
1114 1115 1116
  select_result *result;
  TMP_TABLE_PARAM tmp_table_param;
  MYSQL_LOCK *lock;
unknown's avatar
unknown committed
1117
  /// unit structure (with global parameters) for this select
1118
  SELECT_LEX_UNIT *unit;
unknown's avatar
unknown committed
1119
  /// select that processed
1120
  SELECT_LEX *select_lex;
unknown's avatar
unknown committed
1121
  /** 
1122 1123 1124 1125 1126 1127 1128
    TRUE <=> optimizer must not mark any table as a constant table.
    This is needed for subqueries in form "a IN (SELECT .. UNION SELECT ..):
    when we optimize the select that reads the results of the union from a
    temporary table, we must not mark the temp. table as constant because
    the number of rows in it may vary from one subquery execution to another.
  */
  bool no_const_tables; 
1129 1130 1131 1132 1133
  /*
    This flag is set if we call no_rows_in_result() as par of end_group().
    This is used as a simple speed optimization to avoiding calling
    restore_no_rows_in_result() in ::reinit()
  */
1134
  bool no_rows_in_result_called;
1135 1136 1137 1138 1139 1140

  /**
    This is set if SQL_CALC_ROWS was calculated by filesort()
    and should be taken from the appropriate JOIN_TAB
  */
  bool filesort_found_rows;
1141
  
1142 1143 1144
  /**
    Copy of this JOIN to be used with temporary tables.

1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156
    tmp_join is used when the JOIN needs to be "reusable" (e.g. in a
    subquery that gets re-executed several times) and we know will use
    temporary tables for materialization. The materialization to a
    temporary table overwrites the JOIN structure to point to the
    temporary table after the materialization is done. This is where
    tmp_join is used : it's a copy of the JOIN before the
    materialization and is used in restoring before re-execution by
    overwriting the current JOIN structure with the saved copy.
    Because of this we should pay extra care of not freeing up helper
    structures that are referenced by the original contents of the
    JOIN. We can check for this by making sure the "current" join is
    not the temporary copy, e.g.  !tmp_join || tmp_join != join
1157
 
1158 1159
    We should free these sub-structures at JOIN::destroy() if the
    "current" join has a copy is not that copy.
1160 1161
  */
  JOIN *tmp_join;
unknown's avatar
unknown committed
1162
  ROLLUP rollup;				///< Used with rollup
1163 1164
  
  bool mixed_implicit_grouping;
unknown's avatar
unknown committed
1165 1166
  bool select_distinct;				///< Set if SELECT DISTINCT
  /**
1167 1168 1169 1170 1171 1172 1173
    If we have the GROUP BY statement in the query,
    but the group_list was emptied by optimizer, this
    flag is TRUE.
    It happens when fields in the GROUP BY are from
    constant table
  */
  bool group_optimized_away;
unknown's avatar
unknown committed
1174 1175 1176 1177 1178

  /*
    simple_xxxxx is set if ORDER/GROUP BY doesn't include any references
    to other tables than the first non-constant table in the JOIN.
    It's also set if ORDER/GROUP BY is empty.
1179 1180
    Used for deciding for or against using a temporary table to compute 
    GROUP/ORDER BY.
unknown's avatar
unknown committed
1181 1182
  */
  bool simple_order, simple_group;
unknown's avatar
unknown committed
1183
  /**
unknown's avatar
unknown committed
1184 1185 1186 1187
    Is set only in case if we have a GROUP BY clause
    and no ORDER BY after constant elimination of 'order'.
  */
  bool no_order;
unknown's avatar
unknown committed
1188
  /** Is set if we have a GROUP BY and we have ORDER BY on a constant. */
unknown's avatar
unknown committed
1189 1190
  bool          skip_sort_order;

1191
  bool need_tmp, hidden_group_fields;
1192 1193
  /* TRUE if there was full cleunap of the JOIN */
  bool cleaned;
1194
  DYNAMIC_ARRAY keyuse;
1195
  Item::cond_result cond_value, having_value;
Igor Babaev's avatar
Igor Babaev committed
1196 1197 1198 1199 1200
  /**
    Impossible where after reading const tables 
    (set in make_join_statistics())
  */
  bool impossible_where; 
unknown's avatar
unknown committed
1201 1202
  List<Item> all_fields; ///< to store all fields that used in query
  ///Above list changed to use temporary table
1203
  List<Item> tmp_all_fields1, tmp_all_fields2, tmp_all_fields3;
unknown's avatar
unknown committed
1204
  ///Part, shared with list above, emulate following list
1205
  List<Item> tmp_fields_list1, tmp_fields_list2, tmp_fields_list3;
unknown's avatar
unknown committed
1206
  List<Item> &fields_list; ///< hold field list passed to mysql_select
1207
  List<Item> procedure_fields_list;
1208 1209 1210 1211
  int error;

  ORDER *order, *group_list, *proc_param; //hold parameters of mysql_select
  COND *conds;                            // ---"---
unknown's avatar
unknown committed
1212
  Item *conds_history;                    // store WHERE for explain
Igor Babaev's avatar
Igor Babaev committed
1213
  COND *outer_ref_cond;       ///<part of conds containing only outer references
1214
  COND *pseudo_bits_cond;     // part of conds containing special bita
unknown's avatar
unknown committed
1215 1216
  TABLE_LIST *tables_list;           ///<hold 'tables' parameter of mysql_select
  List<TABLE_LIST> *join_list;       ///< list of joined tables in reverse order
1217
  COND_EQUAL *cond_equal;
1218
  COND_EQUAL *having_equal;
1219 1220 1221 1222 1223 1224
  /*
    Constant codition computed during optimization, but evaluated during
    join execution. Typically expensive conditions that should not be
    evaluated at optimization time.
  */
  Item *exec_const_cond;
unknown's avatar
unknown committed
1225 1226 1227 1228 1229 1230 1231
  /*
    Constant ORDER and/or GROUP expressions that contain subqueries. Such
    expressions need to evaluated to verify that the subquery indeed
    returns a single row. The evaluation of such expressions is delayed
    until query execution.
  */
  List<Item> exec_const_order_group_cond;
unknown's avatar
unknown committed
1232 1233 1234
  SQL_SELECT *select;                ///<created in optimisation phase
  JOIN_TAB *return_tab;              ///<used only for outer joins
  Item **ref_pointer_array; ///<used pointer reference for this select
1235
  // Copy of above to be used with different lists
1236
  Item **items0, **items1, **items2, **items3, **current_ref_pointer_array;
unknown's avatar
unknown committed
1237 1238
  uint ref_pointer_array_size; ///< size of above in bytes
  const char *zero_result_cause; ///< not 0 if exec must return zero result
unknown's avatar
unknown committed
1239
  
unknown's avatar
unknown committed
1240
  bool union_part; ///< this subselect is part of union 
1241 1242 1243 1244

  enum join_optimization_state { NOT_OPTIMIZED=0,
                                 OPTIMIZATION_IN_PROGRESS=1,
                                 OPTIMIZATION_DONE=2};
unknown's avatar
unknown committed
1245
  bool optimized; ///< flag to avoid double optimization in EXPLAIN
1246
  bool initialized; ///< flag to avoid double init_execution calls
1247 1248
  
  enum { QEP_NOT_PRESENT_YET, QEP_AVAILABLE, QEP_DELETED} have_query_plan;
1249

1250 1251 1252 1253 1254 1255
  /*
    Additional WHERE and HAVING predicates to be considered for IN=>EXISTS
    subquery transformation of a JOIN object.
  */
  Item *in_to_exists_where;
  Item *in_to_exists_having;
1256
  
1257 1258
  /* Temporary tables used to weed-out semi-join duplicates */
  List<TABLE> sj_tmp_tables;
Sergey Petrunya's avatar
Sergey Petrunya committed
1259
  /* SJM nests that are executed with SJ-Materialization strategy */
1260 1261
  List<SJ_MATERIALIZATION_INFO> sjm_info_list;

1262 1263 1264 1265 1266 1267
  /* 
    storage for caching buffers allocated during query execution. 
    These buffers allocations need to be cached as the thread memory pool is
    cleared only at the end of the execution of the whole query and not caching
    allocations that occur in repetition at execution time will result in 
    excessive memory usage.
1268 1269 1270
    Note: make_simple_join always creates an execution plan that accesses
    a single table, thus it is sufficient to have a one-element array for
    table_reexec.
1271 1272
  */  
  SORT_FIELD *sortorder;                        // make_unireg_sortorder()
1273
  TABLE *table_reexec[1];                       // make_simple_join()
1274
  JOIN_TAB *join_tab_reexec;                    // make_simple_join()
1275 1276
  /* end of allocation caching storage */

1277
  JOIN(THD *thd_arg, List<Item> &fields_arg, ulonglong select_options_arg,
1278
       select_result *result_arg)
Igor Babaev's avatar
Igor Babaev committed
1279
    :fields_list(fields_arg)
1280
  {
1281
    init(thd_arg, fields_arg, select_options_arg, result_arg);
1282
  }
1283

1284
  void init(THD *thd_arg, List<Item> &fields_arg, ulonglong select_options_arg,
1285 1286
       select_result *result_arg)
  {
1287
    join_tab= join_tab_save= 0;
1288
    table= 0;
1289
    table_count= 0;
1290
    top_join_tab_count= 0;
1291
    const_tables= 0;
Sergey Petrunia's avatar
Sergey Petrunia committed
1292
    eliminated_tables= 0;
1293
    join_list= 0;
1294
    implicit_grouping= FALSE;
1295 1296 1297 1298 1299
    sort_and_group= 0;
    first_record= 0;
    do_send_rows= 1;
    send_records= 0;
    found_records= 0;
1300
    fetch_limit= HA_POS_ERROR;
1301 1302 1303
    examined_rows= 0;
    exec_tmp_table1= 0;
    exec_tmp_table2= 0;
1304
    sortorder= 0;
1305
    table_reexec[0]= 0;
1306
    join_tab_reexec= 0;
1307
    thd= thd_arg;
unknown's avatar
merge  
unknown committed
1308
    sum_funcs= sum_funcs2= 0;
1309
    procedure= 0;
unknown's avatar
unknown committed
1310
    having= tmp_having= having_history= 0;
1311 1312 1313 1314 1315
    select_options= select_options_arg;
    result= result_arg;
    lock= thd_arg->lock;
    select_lex= 0; //for safety
    tmp_join= 0;
1316
    select_distinct= MY_TEST(select_options & SELECT_DISTINCT);
1317 1318 1319
    no_order= 0;
    simple_order= 0;
    simple_group= 0;
Sergei Golubchik's avatar
Sergei Golubchik committed
1320
    need_distinct= 0;
1321 1322 1323 1324 1325
    skip_sort_order= 0;
    need_tmp= 0;
    hidden_group_fields= 0; /*safety*/
    error= 0;
    select= 0;
1326
    return_tab= 0;
1327 1328 1329 1330
    ref_pointer_array= items0= items1= items2= items3= 0;
    ref_pointer_array_size= 0;
    zero_result_cause= 0;
    optimized= 0;
1331
    have_query_plan= QEP_NOT_PRESENT_YET;
1332
    initialized= 0;
1333
    cleaned= 0;
1334
    cond_equal= 0;
1335
    having_equal= 0;
1336
    exec_const_cond= 0;
1337
    group_optimized_away= 0;
1338
    no_rows_in_result_called= 0;
1339
    positions= best_positions= 0;
1340

unknown's avatar
unknown committed
1341
    all_fields= fields_arg;
1342 1343
    if (&fields_list != &fields_arg)      /* Avoid valgrind-warning */
      fields_list= fields_arg;
1344
    bzero((char*) &keyuse,sizeof(keyuse));
1345
    tmp_table_param.init();
1346
    tmp_table_param.end_write_records= HA_POS_ERROR;
1347
    rollup.state= ROLLUP::STATE_NONE;
1348 1349

    no_const_tables= FALSE;
1350
    outer_ref_cond= pseudo_bits_cond= NULL;
1351 1352
    in_to_exists_where= NULL;
    in_to_exists_having= NULL;
1353
    pre_sort_join_tab= NULL;
Igor Babaev's avatar
Igor Babaev committed
1354 1355
    emb_sjm_nest= NULL;
    sjm_lookup_tables= 0;
1356

1357
    filesort_found_rows= false;
Sergey Petrunya's avatar
Sergey Petrunya committed
1358 1359 1360 1361 1362 1363
    /* 
      The following is needed because JOIN::cleanup(true) may be called for 
      joins for which JOIN::optimize was aborted with an error before a proper
      query plan was produced
    */
    table_access_tabs= NULL; 
1364
  }
1365

1366
  int prepare(Item ***rref_pointer_array, TABLE_LIST *tables, uint wind_num,
1367 1368
	      COND *conds, uint og_num, ORDER *order, bool skip_order_by,
              ORDER *group, Item *having, ORDER *proc_param, SELECT_LEX *select,
1369
	      SELECT_LEX_UNIT *unit);
1370
  bool prepare_stage2();
1371
  int optimize();
1372
  int optimize_inner();
unknown's avatar
unknown committed
1373
  int reinit();
1374
  int init_execution();
1375
  void exec();
1376
  void exec_inner();
1377
  int destroy();
1378
  void restore_tmp();
1379
  bool alloc_func_list();
1380
  bool flatten_subqueries();
1381
  bool optimize_unflattened_subqueries();
unknown's avatar
unknown committed
1382
  bool optimize_constant_subqueries();
1383
  bool make_sum_func_list(List<Item> &all_fields, List<Item> &send_fields,
1384
			  bool before_group_by, bool recompute= FALSE);
1385

1386 1387 1388 1389 1390
  inline void set_items_ref_array(Item **ptr)
  {
    memcpy((char*) ref_pointer_array, (char*) ptr, ref_pointer_array_size);
    current_ref_pointer_array= ptr;
  }
1391 1392 1393 1394
  inline void init_items_ref_array()
  {
    items0= ref_pointer_array + all_fields.elements;
    memcpy(items0, ref_pointer_array, ref_pointer_array_size);
1395
    current_ref_pointer_array= items0;
1396
  }
1397 1398

  bool rollup_init();
1399
  bool rollup_process_const_fields();
1400 1401
  bool rollup_make_fields(List<Item> &all_fields, List<Item> &fields,
			  Item_sum ***func);
1402
  int rollup_send_data(uint idx);
unknown's avatar
unknown committed
1403
  int rollup_write_data(uint idx, TABLE *table);
unknown's avatar
unknown committed
1404
  /**
1405 1406 1407 1408 1409
    Release memory and, if possible, the open tables held by this execution
    plan (and nested plans). It's used to release some tables before
    the end of execution in order to increase concurrency and reduce
    memory consumption.
  */
1410
  void join_free();
unknown's avatar
unknown committed
1411
  /** Cleanup this JOIN, possibly for reuse */
1412
  void cleanup(bool full);
1413
  void clear();
1414
  bool save_join_tab();
unknown's avatar
unknown committed
1415
  bool init_save_join_tab();
1416 1417
  bool send_row_on_empty_set()
  {
unknown's avatar
MWL#89  
unknown committed
1418
    return (do_send_rows && implicit_grouping && !group_optimized_away &&
unknown's avatar
unknown committed
1419
            having_value != Item::COND_FALSE);
1420
  }
unknown's avatar
unknown committed
1421
  bool empty_result() { return (zero_result_cause && !implicit_grouping); }
1422
  bool change_result(select_result *new_result, select_result *old_result);
1423 1424 1425 1426 1427
  bool is_top_level_join() const
  {
    return (unit == &thd->lex->unit && (unit->fake_select_lex == 0 ||
                                        select_lex == unit->fake_select_lex));
  }
1428
  void cache_const_exprs();
Sergey Petrunya's avatar
Sergey Petrunya committed
1429 1430
  inline table_map all_tables_map()
  {
1431
    return (table_map(1) << table_count) - 1;
Sergey Petrunya's avatar
Sergey Petrunya committed
1432
  }
1433
  void drop_unused_derived_keys();
Igor Babaev's avatar
Igor Babaev committed
1434
  inline void eval_select_list_used_tables();
1435 1436 1437 1438 1439 1440 1441 1442 1443 1444
  /* 
    Return the table for which an index scan can be used to satisfy 
    the sort order needed by the ORDER BY/(implicit) GROUP BY clause 
  */
  JOIN_TAB *get_sort_by_join_tab()
  {
    return (need_tmp || !sort_by_table || skip_sort_order ||
            ((group || tmp_table_param.sum_func_count) && !group_list)) ?
              NULL : join_tab+const_tables;
  }
unknown's avatar
unknown committed
1445
  bool setup_subquery_caches();
1446 1447 1448
  bool shrink_join_buffers(JOIN_TAB *jt, 
                           ulonglong curr_space,
                           ulonglong needed_space);
1449 1450 1451
  void set_allowed_join_cache_types();
  bool is_allowed_hash_join_access()
  { 
1452
    return MY_TEST(allowed_join_cache_types & JOIN_CACHE_HASHED_BIT) &&
1453 1454
           max_allowed_join_cache_level > JOIN_CACHE_HASHED_BIT;
  }
Igor Babaev's avatar
Igor Babaev committed
1455 1456 1457 1458 1459 1460 1461 1462 1463 1464 1465 1466 1467 1468 1469 1470
  /*
    Check if we need to create a temporary table.
    This has to be done if all tables are not already read (const tables)
    and one of the following conditions holds:
    - We are using DISTINCT (simple distinct's are already optimized away)
    - We are using an ORDER BY or GROUP BY on fields not in the first table
    - We are using different ORDER BY and GROUP BY orders
    - The user wants us to buffer the result.
    When the WITH ROLLUP modifier is present, we cannot skip temporary table
    creation for the DISTINCT clause just because there are only const tables.
  */
  bool test_if_need_tmp_table()
  {
    return ((const_tables != table_count &&
	    ((select_distinct || !simple_order || !simple_group) ||
	     (group_list && order) ||
1471
             MY_TEST(select_options & OPTION_BUFFER_RESULT))) ||
Igor Babaev's avatar
Igor Babaev committed
1472 1473
            (rollup.state != ROLLUP::STATE_NONE && select_distinct));
  }
1474
  bool choose_subquery_plan(table_map join_tables);
1475
  void get_partial_cost_and_fanout(int end_tab_idx,
Sergey Petrunya's avatar
Sergey Petrunya committed
1476 1477 1478
                                   table_map filter_map,
                                   double *read_time_arg, 
                                   double *record_count_arg);
Sergey Petrunya's avatar
Sergey Petrunya committed
1479 1480 1481
  void get_prefix_cost_and_fanout(uint n_tables, 
                                  double *read_time_arg,
                                  double *record_count_arg);
unknown's avatar
unknown committed
1482
  double get_examined_rows();
1483 1484
  /* defined in opt_subselect.cc */
  bool transform_max_min_subquery();
1485 1486 1487 1488 1489
  /* True if this JOIN is a subquery under an IN predicate. */
  bool is_in_subquery()
  {
    return (unit->item && unit->item->is_in_predicate());
  }
1490 1491 1492 1493 1494
  void save_explain_data(Explain_query *output, bool can_overwrite,
                         bool need_tmp_table, bool need_order, bool distinct);
  int save_explain_data_intern(Explain_query *output, bool need_tmp_table,
                               bool need_order, bool distinct,
                               const char *message);
1495
private:
1496 1497 1498 1499 1500
  /**
    TRUE if the query contains an aggregate function but has no GROUP
    BY clause. 
  */
  bool implicit_grouping; 
1501
  bool make_simple_join(JOIN *join, TABLE *tmp_table);
1502
  void cleanup_item_list(List<Item> &items) const;
unknown's avatar
unknown committed
1503 1504
};

1505 1506
enum enum_with_bush_roots { WITH_BUSH_ROOTS, WITHOUT_BUSH_ROOTS};
enum enum_with_const_tables { WITH_CONST_TABLES, WITHOUT_CONST_TABLES};
unknown's avatar
unknown committed
1507

Sergei Golubchik's avatar
Sergei Golubchik committed
1508
JOIN_TAB *first_linear_tab(JOIN *join,
1509 1510
                           enum enum_with_bush_roots include_bush_roots,
                           enum enum_with_const_tables const_tbls);
1511 1512
JOIN_TAB *next_linear_tab(JOIN* join, JOIN_TAB* tab, 
                          enum enum_with_bush_roots include_bush_roots);
unknown's avatar
unknown committed
1513

1514 1515
JOIN_TAB *first_top_level_tab(JOIN *join, enum enum_with_const_tables with_const);
JOIN_TAB *next_top_level_tab(JOIN *join, JOIN_TAB *tab);
unknown's avatar
unknown committed
1516 1517 1518 1519 1520 1521 1522 1523

typedef struct st_select_check {
  uint const_ref,reg_ref;
} SELECT_CHECK;

extern const char *join_type_str[];

/* Extern functions in sql_select.cc */
1524 1525
void count_field_types(SELECT_LEX *select_lex, TMP_TABLE_PARAM *param, 
                       List<Item> &fields, bool reset_with_sum_func);
1526 1527 1528 1529
bool setup_copy_fields(THD *thd, TMP_TABLE_PARAM *param,
		       Item **ref_pointer_array,
		       List<Item> &new_list1, List<Item> &new_list2,
		       uint elements, List<Item> &fields);
unknown's avatar
unknown committed
1530
void copy_fields(TMP_TABLE_PARAM *param);
1531
bool copy_funcs(Item **func_ptr, const THD *thd);
unknown's avatar
unknown committed
1532
uint find_shortest_key(TABLE *table, const key_map *usable_keys);
unknown's avatar
unknown committed
1533
Field* create_tmp_field_from_field(THD *thd, Field* org_field,
unknown's avatar
unknown committed
1534 1535 1536
                                   const char *name, TABLE *table,
                                   Item_field *item, uint convert_blob_length);
                                                                      
1537 1538
bool is_indexed_agg_distinct(JOIN *join, List<Item_field> *out_args);

unknown's avatar
unknown committed
1539
/* functions from opt_sum.cc */
1540
bool simple_pred(Item_func *func_item, Item **args, bool *inv_order);
1541
int opt_sum_query(THD* thd,
Igor Babaev's avatar
Merge  
Igor Babaev committed
1542
                  List<TABLE_LIST> &tables, List<Item> &all_fields, COND *conds);
unknown's avatar
unknown committed
1543

1544
/* from sql_delete.cc, used by opt_range.cc */
1545
extern "C" int refpos_order_cmp(void* arg, const void *a,const void *b);
1546

unknown's avatar
unknown committed
1547
/** class to copying an field/item to a key struct */
unknown's avatar
unknown committed
1548 1549 1550

class store_key :public Sql_alloc
{
1551 1552
public:
  bool null_key; /* TRUE <=> the value of the key has a null part */
unknown's avatar
unknown committed
1553
  enum store_key_result { STORE_KEY_OK, STORE_KEY_FATAL, STORE_KEY_CONV };
Igor Babaev's avatar
Igor Babaev committed
1554
  enum Type { FIELD_STORE_KEY, ITEM_STORE_KEY, CONST_ITEM_STORE_KEY };
1555
  store_key(THD *thd, Field *field_arg, uchar *ptr, uchar *null, uint length)
1556
    :null_key(0), null_ptr(null), err(0)
unknown's avatar
unknown committed
1557
  {
1558 1559
    to_field=field_arg->new_key_field(thd->mem_root, field_arg->table,
                                      ptr, length, null, 1);
unknown's avatar
unknown committed
1560
  }
1561
  store_key(store_key &arg)
unknown's avatar
unknown committed
1562
    :Sql_alloc(), null_key(arg.null_key), to_field(arg.to_field),
1563 1564 1565
             null_ptr(arg.null_ptr), err(arg.err)

  {}
unknown's avatar
unknown committed
1566
  virtual ~store_key() {}			/** Not actually needed */
Igor Babaev's avatar
Igor Babaev committed
1567
  virtual enum Type type() const=0;
unknown's avatar
unknown committed
1568
  virtual const char *name() const=0;
Igor Babaev's avatar
Igor Babaev committed
1569
  virtual bool store_key_is_const() { return false; }
1570 1571 1572 1573 1574 1575 1576 1577 1578 1579

  /**
    @brief sets ignore truncation warnings mode and calls the real copy method

    @details this function makes sure truncation warnings when preparing the
    key buffers don't end up as errors (because of an enclosing INSERT/UPDATE).
  */
  enum store_key_result copy()
  {
    enum store_key_result result;
1580 1581
    THD *thd= to_field->table->in_use;
    enum_check_fields saved_count_cuted_fields= thd->count_cuted_fields;
1582
    ulonglong sql_mode= thd->variables.sql_mode;
1583
    thd->variables.sql_mode&= ~(MODE_NO_ZERO_IN_DATE | MODE_NO_ZERO_DATE);
1584
    thd->variables.sql_mode|= MODE_INVALID_DATES;
1585

1586
    thd->count_cuted_fields= CHECK_FIELD_IGNORE;
1587 1588 1589

    result= copy_inner();

1590 1591
    thd->count_cuted_fields= saved_count_cuted_fields;
    thd->variables.sql_mode= sql_mode;
1592 1593 1594 1595 1596 1597

    return result;
  }

 protected:
  Field *to_field;				// Store data here
1598 1599
  uchar *null_ptr;
  uchar err;
1600 1601

  virtual enum store_key_result copy_inner()=0;
unknown's avatar
unknown committed
1602 1603 1604 1605 1606 1607 1608 1609
};


class store_key_field: public store_key
{
  Copy_field copy_field;
  const char *field_name;
 public:
1610 1611
  store_key_field(THD *thd, Field *to_field_arg, uchar *ptr,
                  uchar *null_ptr_arg,
unknown's avatar
unknown committed
1612
		  uint length, Field *from_field, const char *name_arg)
unknown's avatar
unknown committed
1613
    :store_key(thd, to_field_arg,ptr,
unknown's avatar
unknown committed
1614
	       null_ptr_arg ? null_ptr_arg : from_field->maybe_null() ? &err
1615
	       : (uchar*) 0, length), field_name(name_arg)
unknown's avatar
unknown committed
1616 1617 1618 1619 1620
  {
    if (to_field)
    {
      copy_field.set(to_field,from_field,0);
    }
Igor Babaev's avatar
Igor Babaev committed
1621 1622 1623
  }  

  enum Type type() const { return FIELD_STORE_KEY; }
1624 1625
  const char *name() const { return field_name; }

Igor Babaev's avatar
Igor Babaev committed
1626 1627 1628 1629 1630 1631
  void change_source_field(Item_field *fld_item)
  {
    copy_field.set(to_field, fld_item->field, 0);
    field_name= fld_item->full_name();
  }

1632 1633
 protected: 
  enum store_key_result copy_inner()
1634
  {
1635 1636 1637
    TABLE *table= copy_field.to_field->table;
    my_bitmap_map *old_map= dbug_tmp_use_all_columns(table,
                                                     table->write_set);
Igor Babaev's avatar
Igor Babaev committed
1638 1639 1640 1641 1642 1643 1644 1645 1646

    /* 
      It looks like the next statement is needed only for a simplified
      hash function over key values used now in BNLH join.
      When the implementation of this function will be replaced for a proper
      full version this statement probably should be removed.
    */  
    bzero(copy_field.to_ptr,copy_field.to_length);

1647
    copy_field.do_copy(&copy_field);
1648
    dbug_tmp_restore_column_map(table->write_set, old_map);
1649
    null_key= to_field->is_null();
unknown's avatar
unknown committed
1650
    return err != 0 ? STORE_KEY_FATAL : STORE_KEY_OK;
1651
  }
unknown's avatar
unknown committed
1652 1653 1654 1655 1656 1657 1658
};


class store_key_item :public store_key
{
 protected:
  Item *item;
unknown's avatar
unknown committed
1659 1660 1661 1662 1663
  /*
    Flag that forces usage of save_val() method which save value of the
    item instead of save_in_field() method which saves result.
  */
  bool use_value;
unknown's avatar
unknown committed
1664
public:
1665
  store_key_item(THD *thd, Field *to_field_arg, uchar *ptr,
unknown's avatar
unknown committed
1666
                 uchar *null_ptr_arg, uint length, Item *item_arg, bool val)
1667
    :store_key(thd, to_field_arg, ptr,
unknown's avatar
unknown committed
1668
	       null_ptr_arg ? null_ptr_arg : item_arg->maybe_null ?
unknown's avatar
unknown committed
1669
	       &err : (uchar*) 0, length), item(item_arg), use_value(val)
unknown's avatar
unknown committed
1670
  {}
1671 1672 1673 1674
  store_key_item(store_key &arg, Item *new_item, bool val)
    :store_key(arg), item(new_item), use_value(val)
  {}

Igor Babaev's avatar
Igor Babaev committed
1675 1676

  enum Type type() const { return ITEM_STORE_KEY; }
1677 1678 1679 1680
  const char *name() const { return "func"; }

 protected:  
  enum store_key_result copy_inner()
unknown's avatar
unknown committed
1681
  {
1682 1683 1684
    TABLE *table= to_field->table;
    my_bitmap_map *old_map= dbug_tmp_use_all_columns(table,
                                                     table->write_set);
unknown's avatar
unknown committed
1685
    int res= FALSE;
Igor Babaev's avatar
Igor Babaev committed
1686 1687 1688 1689 1690 1691 1692 1693 1694

    /* 
      It looks like the next statement is needed only for a simplified
      hash function over key values used now in BNLH join.
      When the implementation of this function will be replaced for a proper
      full version this statement probably should be removed.
    */  
    to_field->reset();

unknown's avatar
unknown committed
1695 1696 1697 1698
    if (use_value)
      item->save_val(to_field);
    else
      res= item->save_in_field(to_field, 1);
1699 1700 1701 1702 1703
    /*
     Item::save_in_field() may call Item::val_xxx(). And if this is a subquery
     we need to check for errors executing it and react accordingly
    */
    if (!res && table->in_use->is_error())
1704
      res= 1; /* STORE_KEY_FATAL */
1705
    dbug_tmp_restore_column_map(table->write_set, old_map);
1706
    null_key= to_field->is_null() || item->null_value;
1707 1708
    return ((err != 0 || res < 0 || res > 2) ? STORE_KEY_FATAL : 
            (store_key_result) res);
unknown's avatar
unknown committed
1709 1710 1711 1712 1713 1714 1715 1716
  }
};


class store_key_const_item :public store_key_item
{
  bool inited;
public:
1717 1718
  store_key_const_item(THD *thd, Field *to_field_arg, uchar *ptr,
		       uchar *null_ptr_arg, uint length,
unknown's avatar
unknown committed
1719
		       Item *item_arg)
1720
    :store_key_item(thd, to_field_arg, ptr,
unknown's avatar
unknown committed
1721
		    null_ptr_arg ? null_ptr_arg : item_arg->maybe_null ?
unknown's avatar
unknown committed
1722
		    &err : (uchar*) 0, length, item_arg, FALSE), inited(0)
unknown's avatar
unknown committed
1723 1724
  {
  }
1725 1726 1727
  store_key_const_item(store_key &arg, Item *new_item)
    :store_key_item(arg, new_item, FALSE), inited(0)
  {}
Igor Babaev's avatar
Igor Babaev committed
1728 1729

  enum Type type() const { return CONST_ITEM_STORE_KEY; }
1730
  const char *name() const { return "const"; }
Igor Babaev's avatar
Igor Babaev committed
1731
  bool store_key_is_const() { return true; }
1732 1733 1734

protected:  
  enum store_key_result copy_inner()
unknown's avatar
unknown committed
1735
  {
unknown's avatar
unknown committed
1736
    int res;
unknown's avatar
unknown committed
1737 1738 1739
    if (!inited)
    {
      inited=1;
1740 1741 1742
      TABLE *table= to_field->table;
      my_bitmap_map *old_map= dbug_tmp_use_all_columns(table,
                                                       table->write_set);
unknown's avatar
unknown committed
1743 1744 1745
      if ((res= item->save_in_field(to_field, 1)))
      {       
        if (!err)
1746
          err= res < 0 ? 1 : res; /* 1=STORE_KEY_FATAL */
unknown's avatar
unknown committed
1747
      }
1748 1749 1750 1751 1752
      /*
        Item::save_in_field() may call Item::val_xxx(). And if this is a subquery
        we need to check for errors executing it and react accordingly
        */
      if (!err && to_field->table->in_use->is_error())
1753
        err= 1; /* STORE_KEY_FATAL */
1754
      dbug_tmp_restore_column_map(table->write_set, old_map);
unknown's avatar
unknown committed
1755
    }
1756
    null_key= to_field->is_null() || item->null_value;
1757
    return (err > 2 ? STORE_KEY_FATAL : (store_key_result) err);
unknown's avatar
unknown committed
1758 1759 1760
  }
};

1761
bool cp_buffer_from_ref(THD *thd, TABLE *table, TABLE_REF *ref);
unknown's avatar
unknown committed
1762
bool error_if_full_join(JOIN *join);
1763
int report_error(TABLE *table, int error);
1764
int safe_index_read(JOIN_TAB *tab);
1765
COND *remove_eq_conds(THD *thd, COND *cond, Item::cond_result *cond_value);
1766 1767 1768 1769 1770 1771 1772 1773 1774
int get_quick_record(SQL_SELECT *select);
SORT_FIELD * make_unireg_sortorder(ORDER *order, uint *length,
                                  SORT_FIELD *sortorder);
int setup_order(THD *thd, Item **ref_pointer_array, TABLE_LIST *tables,
		List<Item> &fields, List <Item> &all_fields, ORDER *order);
int setup_group(THD *thd, Item **ref_pointer_array, TABLE_LIST *tables,
		List<Item> &fields, List<Item> &all_fields, ORDER *order,
		bool *hidden_group_fields);
bool fix_inner_refs(THD *thd, List<Item> &all_fields, SELECT_LEX *select,
Sergei Golubchik's avatar
merge.  
Sergei Golubchik committed
1775 1776 1777
                   Item **ref_pointer_array);
int join_read_key2(THD *thd, struct st_join_table *tab, TABLE *table,
                   struct st_table_ref *table_ref);
1778 1779 1780 1781 1782 1783 1784 1785 1786 1787 1788 1789 1790 1791 1792 1793 1794 1795 1796

bool handle_select(THD *thd, LEX *lex, select_result *result,
                   ulong setup_tables_done_option);
bool mysql_select(THD *thd, Item ***rref_pointer_array,
                  TABLE_LIST *tables, uint wild_num,  List<Item> &list,
                  COND *conds, uint og_num, ORDER *order, ORDER *group,
                  Item *having, ORDER *proc_param, ulonglong select_type, 
                  select_result *result, SELECT_LEX_UNIT *unit, 
                  SELECT_LEX *select_lex);
void free_underlaid_joins(THD *thd, SELECT_LEX *select);
bool mysql_explain_union(THD *thd, SELECT_LEX_UNIT *unit,
                         select_result *result);
Field *create_tmp_field(THD *thd, TABLE *table,Item *item, Item::Type type,
			Item ***copy_func, Field **from_field,
                        Field **def_field,
			bool group, bool modify_item,
			bool table_cant_handle_bit_fields,
                        bool make_copy_field,
                        uint convert_blob_length);
Sergei Golubchik's avatar
Sergei Golubchik committed
1797
bool create_internal_tmp_table(TABLE *table, KEY *keyinfo, 
Sergei Golubchik's avatar
Sergei Golubchik committed
1798 1799
                               TMP_ENGINE_COLUMNDEF *start_recinfo,
                               TMP_ENGINE_COLUMNDEF **recinfo, 
Sergei Golubchik's avatar
Sergei Golubchik committed
1800
                               ulonglong options, my_bool big_tables);
1801 1802 1803 1804 1805 1806 1807 1808 1809 1810 1811

/*
  General routine to change field->ptr of a NULL-terminated array of Field
  objects. Useful when needed to call val_int, val_str or similar and the
  field data is not in table->record[0] but in some other structure.
  set_key_field_ptr changes all fields of an index using a key_info object.
  All methods presume that there is at least one field to change.
*/

TABLE *create_virtual_tmp_table(THD *thd, List<Create_field> &field_list);

1812
int test_if_item_cache_changed(List<Cached_item> &list);
1813
int join_init_read_record(JOIN_TAB *tab);
1814
int join_read_record_no_init(JOIN_TAB *tab);
Sergey Petrunia's avatar
Sergey Petrunia committed
1815
void set_position(JOIN *join,uint idx,JOIN_TAB *table,KEYUSE *key);
1816 1817 1818 1819
inline Item * and_items(Item* cond, Item *item)
{
  return (cond? (new Item_cond_and(cond, item)) : item);
}
unknown's avatar
MWL#89  
unknown committed
1820
bool choose_plan(JOIN *join, table_map join_tables);
1821 1822 1823
void optimize_wo_join_buffering(JOIN *join, uint first_tab, uint last_tab, 
                                table_map last_remaining_tables, 
                                bool first_alt, uint no_jbuf_before,
1824
                                double *outer_rec_count, double *reopt_cost);
1825 1826
Item_equal *find_item_equal(COND_EQUAL *cond_equal, Field *field,
                            bool *inherited_fl);
1827
extern bool test_if_ref(Item *, 
1828
                 Item_field *left_item,Item *right_item);
1829 1830 1831 1832 1833 1834

inline bool optimizer_flag(THD *thd, uint flag)
{ 
  return (thd->variables.optimizer_switch & flag);
}

1835
/*
Sergey Petrunya's avatar
Sergey Petrunya committed
1836 1837
int print_fake_select_lex_join(select_result_sink *result, bool on_the_fly,
                               SELECT_LEX *select_lex, uint8 select_options);
1838
*/
Sergey Petrunya's avatar
Sergey Petrunya committed
1839

1840
uint get_index_for_order(ORDER *order, TABLE *table, SQL_SELECT *select,
1841 1842
                         ha_rows limit, ha_rows *scanned_limit, 
                         bool *need_sort, bool *reverse);
1843 1844 1845 1846
ORDER *simple_remove_const(ORDER *order, COND *where);
bool const_expression_in_where(COND *cond, Item *comp_item,
                               Field *comp_field= NULL,
                               Item **const_item= NULL);
1847
bool cond_is_datetime_is_null(Item *cond);
1848
bool cond_has_datetime_is_null(Item *cond);
1849

1850
/* Table elimination entry point function */
Sergey Petrunya's avatar
Sergey Petrunya committed
1851
void eliminate_tables(JOIN *join);
Sergey Petrunia's avatar
Sergey Petrunia committed
1852

1853
/* Index Condition Pushdown entry point function */
1854
void push_index_cond(JOIN_TAB *tab, uint keyno);
1855

Igor Babaev's avatar
Igor Babaev committed
1856 1857
#define OPT_LINK_EQUAL_FIELDS    1

Sergey Petrunya's avatar
Sergey Petrunya committed
1858 1859
/* EXPLAIN-related utility functions */
int print_explain_message_line(select_result_sink *result, 
Sergei Petrunia's avatar
Sergei Petrunia committed
1860
                               uint8 options, bool is_analyze,
Sergey Petrunya's avatar
Sergey Petrunya committed
1861 1862
                               uint select_number,
                               const char *select_type,
1863
                               ha_rows *rows,
Sergey Petrunya's avatar
Sergey Petrunya committed
1864 1865
                               const char *message);
void explain_append_mrr_info(QUICK_RANGE_SELECT *quick, String *res);
1866 1867
int append_possible_keys(MEM_ROOT *alloc, String_list &list, TABLE *table, 
                         key_map possible_keys);
Sergey Petrunya's avatar
Sergey Petrunya committed
1868

1869 1870 1871 1872 1873 1874 1875 1876 1877 1878 1879 1880
/****************************************************************************
  Temporary table support for SQL Runtime
 ***************************************************************************/

#define STRING_TOTAL_LENGTH_TO_PACK_ROWS 128
#define AVG_STRING_LENGTH_TO_PACK_ROWS   64
#define RATIO_TO_PACK_ROWS	       2
#define MIN_STRING_LENGTH_TO_PACK_ROWS   10

TABLE *create_tmp_table(THD *thd,TMP_TABLE_PARAM *param,List<Item> &fields,
			ORDER *group, bool distinct, bool save_sum_fields,
			ulonglong select_options, ha_rows rows_limit,
1881 1882
			const char* alias, bool do_not_open=FALSE,
                        bool keep_row_order= FALSE);
1883 1884
void free_tmp_table(THD *thd, TABLE *entry);
bool create_internal_tmp_table_from_heap(THD *thd, TABLE *table,
Sergei Golubchik's avatar
Sergei Golubchik committed
1885 1886
                                         TMP_ENGINE_COLUMNDEF *start_recinfo,
                                         TMP_ENGINE_COLUMNDEF **recinfo, 
1887 1888
                                         int error, bool ignore_last_dupp_key_error,
                                         bool *is_duplicate);
1889
bool create_internal_tmp_table(TABLE *table, KEY *keyinfo, 
Sergei Golubchik's avatar
Sergei Golubchik committed
1890 1891
                               TMP_ENGINE_COLUMNDEF *start_recinfo,
                               TMP_ENGINE_COLUMNDEF **recinfo, 
1892
                               ulonglong options);
1893 1894
bool open_tmp_table(TABLE *table);
void setup_tmp_table_column_bitmaps(TABLE *table, uchar *bitmaps);
unknown's avatar
MWL#89  
unknown committed
1895
double prev_record_reads(POSITION *positions, uint idx, table_map found_ref);
1896
void fix_list_after_tbl_changes(SELECT_LEX *new_parent, List<TABLE_LIST> *tlist);
1897

unknown's avatar
unknown committed
1898 1899 1900 1901 1902 1903 1904 1905 1906 1907 1908 1909 1910
struct st_cond_statistic
{
  Item *cond;
  Field *field_arg;
  ulong positive;
};
typedef struct st_cond_statistic COND_STATISTIC;

ulong check_selectivity(THD *thd,
                        ulong rows_to_read,
                        TABLE *table,
                        List<COND_STATISTIC> *conds);

Sergei Petrunia's avatar
Sergei Petrunia committed
1911

1912
#endif /* SQL_SELECT_INCLUDED */