#ifndef SQL_PLANNER_INCLUDED #define SQL_PLANNER_INCLUDED /* Copyright (c) 2000, 2012, Oracle and/or its affiliates. All rights reserved. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License, version 2.0, as published by the Free Software Foundation. This program is also distributed with certain software (including but not limited to OpenSSL) that is licensed under separate terms, as designated in a particular file or component or in included license documentation. The authors of MySQL hereby grant you an additional permission to link the program and your derivative works with the separately licensed software that they have included with MySQL. 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, version 2.0, for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ /** @file Join planner classes */ #include "sql_class.h" #include "sql_select.h" #include "sql_test.h" #include "sql_optimizer.h" class Opt_trace_object; /** This class determines the optimal join order for tables within a basic query block, ie a query specification clause, possibly extended with semi-joined tables from embedded subqueries. This class takes as prerequisite a join class where all dependencies among tables have been sorted out, all possible access paths have been sorted out, and all statistics information has been filled in. The class has a sole public function that will calculate the most optimal plan based on the inputs and the environment, such as prune level and greedy optimizer search depth. For more information, see the function headers for the private functions greedy_search(), best_extension_by_limited_search() and eq_ref_extension_by_limited_search(). */ class Optimize_table_order { public: Optimize_table_order(THD *thd, JOIN *join, TABLE_LIST *sjm_nest) : search_depth(determine_search_depth(thd->variables.optimizer_search_depth, join->tables - join->const_tables)), prune_level(thd->variables.optimizer_prune_level), thd(thd), join(join), cur_embedding_map(0), emb_sjm_nest(sjm_nest), excluded_tables((sjm_nest ? (join->all_table_map & ~sjm_nest->sj_inner_tables) : 0) | (join->allow_outer_refs ? 0 : OUTER_REF_TABLE_BIT)) {} ~Optimize_table_order() {} /** Entry point to table join order optimization. For further description, see class header and private function headers. @return false if successful, true if error */ bool choose_table_order(); private: const uint search_depth; // Maximum search depth to apply in greedy search const uint prune_level; // pruning heuristics to be applied // (0 = EXHAUSTIVE, 1 = PRUNE_BY_TIME_OR_ROWS) THD *const thd; // Pointer to current THD JOIN *const join; // Pointer to the current plan being developed /** Bitmap of all join nests embedding the last table appended to the current partial join. */ nested_join_map cur_embedding_map; /** If non-NULL, we are optimizing a materialized semi-join nest. If NULL, we are optimizing a complete join plan. */ const TABLE_LIST *const emb_sjm_nest; /** When calculating a plan for a materialized semi-join nest, best_access_path() needs to know not only the remaining tables within the semi-join nest, but also all tables outside of this nest, because there may be key references between the semi-join nest and the outside tables that should not be considered when materializing the semi-join nest. @c excluded_tables tracks these tables. */ const table_map excluded_tables; void best_access_path(JOIN_TAB *s, table_map remaining_tables, uint idx, bool disable_jbuf, double record_count, POSITION *pos, POSITION *loose_scan_pos); bool check_interleaving_with_nj(JOIN_TAB *next_tab); void advance_sj_state(table_map remaining_tables, const JOIN_TAB *tab, uint idx, double *current_rowcount, double *current_cost, POSITION *loose_scan_pos); void backout_nj_state(const table_map remaining_tables, const JOIN_TAB *tab); void optimize_straight_join(table_map join_tables); bool greedy_search(table_map remaining_tables); bool best_extension_by_limited_search(table_map remaining_tables, uint idx, double record_count, double read_time, uint current_search_depth); table_map eq_ref_extension_by_limited_search( table_map remaining_tables, uint idx, double record_count, double read_time, uint current_search_depth); void consider_plan(uint idx, double record_count, double read_time, Opt_trace_object *trace_obj); bool fix_semijoin_strategies(); bool semijoin_firstmatch_loosescan_access_paths( uint first_tab, uint last_tab, table_map remaining_tables, bool loosescan, bool final, double *newcount, double *newcost); void semijoin_mat_scan_access_paths( uint last_inner_tab, uint last_outer_tab, table_map remaining_tables, TABLE_LIST *sjm_nest, bool final, double *newcount, double *newcost); void semijoin_mat_lookup_access_paths( uint last_inner, TABLE_LIST *sjm_nest, double *newcount, double *newcost); void semijoin_dupsweedout_access_paths( uint first_tab, uint last_tab, table_map remaining_tables, double *newcount, double *newcost); static uint determine_search_depth(uint search_depth, uint table_count); }; void get_partial_join_cost(JOIN *join, uint n_tables, double *read_time_arg, double *record_count_arg); #endif /* SQL_PLANNER_INCLUDED */