30
#include "drizzled/server_includes.h"
31
#include "drizzled/table_map_iterator.h"
35
32
#include "drizzled/item/cache.h"
36
33
#include "drizzled/item/cmpfunc.h"
37
34
#include "drizzled/item/copy_string.h"
47
44
#include "drizzled/field/blob.h"
48
45
#include "drizzled/optimizer/position.h"
49
46
#include "drizzled/optimizer/sargable_param.h"
50
#include "drizzled/optimizer/key_use.h"
51
#include "drizzled/optimizer/range.h"
52
#include "drizzled/optimizer/sum.h"
53
#include "drizzled/optimizer/explain_plan.h"
54
#include "drizzled/records.h"
55
#include "drizzled/probes.h"
56
#include "drizzled/internal/my_bit.h"
57
#include "drizzled/internal/my_sys.h"
58
#include "drizzled/internal/iocache.h"
47
#include "mysys/my_bit.h"
60
49
#include <algorithm>
62
51
using namespace std;
67
extern plugin::StorageEngine *heap_engine;
68
extern std::bitset<12> test_flags;
52
using namespace drizzled;
70
54
/** Declarations of static functions used in this source file. */
71
55
static bool make_group_fields(JOIN *main_join, JOIN *curr_join);
74
58
static uint32_t cache_record_length(JOIN *join, uint32_t index);
75
59
static double prev_record_reads(JOIN *join, uint32_t idx, table_map found_ref);
76
60
static bool get_best_combination(JOIN *join);
77
static void set_position(JOIN *join,
80
optimizer::KeyUse *key);
61
static void set_position(JOIN *join,uint32_t index,JoinTable *table,KeyUse *key);
81
62
static bool choose_plan(JOIN *join,table_map join_tables);
82
63
static void best_access_path(JOIN *join, JoinTable *s,
97
78
static uint32_t determine_search_depth(JOIN* join);
98
79
static bool make_simple_join(JOIN *join,Table *tmp_table);
99
80
static void make_outerjoin_info(JOIN *join);
100
static bool make_join_select(JOIN *join, optimizer::SqlSelect *select,COND *item);
81
static bool make_join_select(JOIN *join,SQL_SELECT *select,COND *item);
101
82
static bool make_join_readinfo(JOIN *join, uint64_t options, uint32_t no_jbuf_after);
102
83
static void update_depend_map(JOIN *join);
103
84
static void update_depend_map(JOIN *join, order_st *order);
199
180
select_lex->leaf_tables, fields_list,
200
181
all_fields, &conds, order, group_list,
201
182
&hidden_group_fields))
183
return(-1); /* purecov: inspected */
204
185
ref_pointer_array= *rref_pointer_array;
231
212
in_subs= (Item_in_subselect*)subselect;
234
bool do_materialize= true;
215
bool do_materialize= !test(session->variables.optimizer_switch &
216
OPTIMIZER_SWITCH_NO_MATERIALIZATION);
236
218
Check if the subquery predicate can be executed via materialization.
237
219
The required conditions are:
249
231
(Subquery is non-correlated ||
250
232
Subquery is correlated to any query outer to IN predicate ||
251
233
(Subquery is correlated to the immediate outer query &&
252
Subquery !contains {GROUP BY, ORDER BY [LIMIT],
234
Subquery !contains {GROUP BY, order_st BY [LIMIT],
253
235
aggregate functions) && subquery predicate is not under "NOT IN"))
254
236
6. No execution method was already chosen (by a prepared statement).
320
goto err; /* purecov: inspected */
341
* The below will create the new table for
342
* CREATE TABLE ... SELECT
344
* @see create_table_from_items() in drizzled/sql_insert.cc
346
322
if (result && result->prepare(fields_list, unit_arg))
323
goto err; /* purecov: inspected */
349
325
/* Init join struct */
350
326
count_field_types(select_lex, &tmp_table_param, all_fields, 0);
507
optimizer::sum_query() returns HA_ERR_KEY_NOT_FOUND if no rows match
483
opt_sum_query() returns HA_ERR_KEY_NOT_FOUND if no rows match
508
484
to the WHERE conditions,
509
485
or 1 if all items were resolved,
510
486
or 0, or an error number HA_ERR_...
512
if ((res= optimizer::sum_query(select_lex->leaf_tables, all_fields, conds)))
488
if ((res=opt_sum_query(select_lex->leaf_tables, all_fields, conds)))
514
490
if (res == HA_ERR_KEY_NOT_FOUND)
532
508
tables_list= 0; // All tables resolved
534
510
Extract all table-independent conditions and replace the WHERE
535
clause with them. All other conditions were computed by optimizer::sum_query
511
clause with them. All other conditions were computed by opt_sum_query
536
512
and the MIN/MAX/COUNT function(s) have been replaced by constants,
537
513
so there is no need to compute the whole WHERE clause again.
538
514
Notice that make_cond_for_table() will always succeed to remove all
539
computed conditions, because optimizer::sum_query() is applicable only to
515
computed conditions, because opt_sum_query() is applicable only to
541
517
Preserve conditions for EXPLAIN.
583
559
if (!(session->options & OPTION_BIG_SELECTS) &&
584
560
best_read > (double) session->variables.max_join_size &&
585
561
!(select_options & SELECT_DESCRIBE))
562
{ /* purecov: inspected */
587
563
my_message(ER_TOO_BIG_SELECT, ER(ER_TOO_BIG_SELECT), MYF(0));
595
571
/* Handle the case where we have an OUTER JOIN without a WHERE */
596
572
conds=new Item_int((int64_t) 1,1); // Always true
598
select= optimizer::make_select(*table, const_table_map,
599
const_table_map, conds, 1, &error);
574
select= make_select(*table, const_table_map,
575
const_table_map, conds, 1, &error);
577
{ /* purecov: inspected */
578
error= -1; /* purecov: inspected */
679
654
The FROM clause must contain a single non-constant table.
681
656
if (tables - const_tables == 1 && (group_list || select_distinct) &&
682
! tmp_table_param.sum_func_count &&
683
(! join_tab[const_tables].select ||
684
! join_tab[const_tables].select->quick ||
657
!tmp_table_param.sum_func_count &&
658
(!join_tab[const_tables].select ||
659
!join_tab[const_tables].select->quick ||
685
660
join_tab[const_tables].select->quick->get_type() !=
686
optimizer::QuickSelectInterface::QS_TYPE_GROUP_MIN_MAX))
661
QUICK_SELECT_I::QS_TYPE_GROUP_MIN_MAX))
688
663
if (group_list && list_contains_unique_index(join_tab[const_tables].table, find_field_in_order_list, (void *) group_list))
691
666
We have found that grouping can be removed since groups correspond to
692
667
only one row anyway, but we still have to guarantee correct result
693
668
order. The line below effectively rewrites the query from GROUP BY
694
<fields> to ORDER BY <fields>. There are two exceptions:
669
<fields> to order_st BY <fields>. There are two exceptions:
695
670
- if skip_sort_order is set (see above), then we can simply skip
697
- we can only rewrite ORDER BY if the ORDER BY fields are 'compatible'
672
- we can only rewrite order_st BY if the order_st BY fields are 'compatible'
698
673
with the GROUP BY ones, i.e. either one is a prefix of another.
699
We only check if the ORDER BY is a prefix of GROUP BY. In this case
674
We only check if the order_st BY is a prefix of GROUP BY. In this case
700
675
test_if_subpart() copies the ASC/DESC attributes from the original
702
677
If GROUP BY is a prefix of order_st BY, then it is safe to leave
705
if (! order || test_if_subpart(group_list, order))
680
if (!order || test_if_subpart(group_list, order))
706
681
order= skip_sort_order ? 0 : group_list;
708
683
If we have an IGNORE INDEX FOR GROUP BY(fields) clause, this must be
737
712
- We are scanning the whole table without LIMIT
738
713
This can happen if:
739
714
- We are using CALC_FOUND_ROWS
740
- We are using an ORDER BY that can't be optimized away.
715
- We are using an order_st BY that can't be optimized away.
742
717
We don't want to use this optimization when we are using LIMIT
743
718
because in this case we can just create a temporary table that
826
801
This has to be done if all tables are not already read (const tables)
827
802
and one of the following conditions holds:
828
803
- We are using DISTINCT (simple distinct's are already optimized away)
829
- We are using an ORDER BY or GROUP BY on fields not in the first table
830
- We are using different ORDER BY and GROUP BY orders
804
- We are using an order_st BY or GROUP BY on fields not in the first table
805
- We are using different order_st BY and GROUP BY orders
831
806
- The user wants us to buffer the result.
833
808
need_tmp= (const_tables != tables &&
1008
980
Pushing LIMIT to the temporary table creation is not applicable
1009
when there is ORDER BY or GROUP BY or there is no GROUP BY, but
981
when there is order_st BY or GROUP BY or there is no GROUP BY, but
1010
982
there are aggregate functions, because in all these cases we need
1011
983
all result rows.
1134
1106
if (exec_tmp_table1)
1136
exec_tmp_table1->cursor->extra(HA_EXTRA_RESET_STATE);
1137
exec_tmp_table1->cursor->ha_delete_all_rows();
1108
exec_tmp_table1->file->extra(HA_EXTRA_RESET_STATE);
1109
exec_tmp_table1->file->ha_delete_all_rows();
1138
1110
exec_tmp_table1->free_io_cache();
1139
1111
exec_tmp_table1->filesort_free_buffers();
1141
1113
if (exec_tmp_table2)
1143
exec_tmp_table2->cursor->extra(HA_EXTRA_RESET_STATE);
1144
exec_tmp_table2->cursor->ha_delete_all_rows();
1115
exec_tmp_table2->file->extra(HA_EXTRA_RESET_STATE);
1116
exec_tmp_table2->file->ha_delete_all_rows();
1145
1117
exec_tmp_table2->free_io_cache();
1146
1118
exec_tmp_table2->filesort_free_buffers();
1219
1191
/* Only test of functions */
1220
1192
if (select_options & SELECT_DESCRIBE)
1222
optimizer::ExplainPlan planner(this,
1226
(zero_result_cause ? zero_result_cause : "No tables used"));
1227
planner.printPlan();
1193
select_describe(this, false, false, false, (zero_result_cause?zero_result_cause:"No tables used"));
1231
1196
result->send_fields(*columns_list);
1244
if ((this->select_lex->options & OPTION_SCHEMA_TABLE) && get_schema_tables_result(this, PROCESSED_BY_JOIN_EXEC))
1279
1247
if (select_options & SELECT_DESCRIBE)
1282
Check if we managed to optimize ORDER BY away and don't use temporary
1250
Check if we managed to optimize order_st BY away and don't use temporary
1283
1251
table to resolve order_st BY: in that case, we only may need to do
1284
1252
filesort for GROUP BY.
1300
1268
having= tmp_having;
1301
optimizer::ExplainPlan planner(this,
1303
order != 0 && ! skip_sort_order,
1305
! tables ? "No tables used" : NULL);
1306
planner.printPlan();
1269
select_describe(this, need_tmp, order != 0 && !skip_sort_order, select_distinct, !tables ? "No tables used" : NULL);
2360
2323
item->save_in_result_field(1);
2362
2325
copy_sum_funcs(sum_funcs_end[i+1], sum_funcs_end[i]);
2363
if ((write_error= table_arg->cursor->ha_write_row(table_arg->record[0])))
2326
if ((write_error= table_arg->file->ha_write_row(table_arg->record[0])))
2365
2328
if (create_myisam_from_heap(session, table_arg,
2366
2329
tmp_table_param.start_recinfo,
2416
Cache constant expressions in WHERE, HAVING, ON conditions.
2419
void JOIN::cache_const_exprs()
2421
bool cache_flag= false;
2422
bool *analyzer_arg= &cache_flag;
2424
/* No need in cache if all tables are constant. */
2425
if (const_tables == tables)
2429
conds->compile(&Item::cache_const_expr_analyzer, (unsigned char **)&analyzer_arg,
2430
&Item::cache_const_expr_transformer, (unsigned char *)&cache_flag);
2433
having->compile(&Item::cache_const_expr_analyzer, (unsigned char **)&analyzer_arg,
2434
&Item::cache_const_expr_transformer, (unsigned char *)&cache_flag);
2436
for (JoinTable *tab= join_tab + const_tables; tab < join_tab + tables ; tab++)
2438
if (*tab->on_expr_ref)
2441
(*tab->on_expr_ref)->compile(&Item::cache_const_expr_analyzer,
2442
(unsigned char **)&analyzer_arg,
2443
&Item::cache_const_expr_transformer,
2444
(unsigned char *)&cache_flag);
2452
2381
Process one record of the nested loop join.
2470
2399
if (join->session->killed) // Aborted by user
2472
2401
join->session->send_kill_message();
2473
return NESTED_LOOP_KILLED;
2402
return NESTED_LOOP_KILLED; /* purecov: inspected */
2475
2404
if (!select_cond || select_cond->val_int())
2675
2604
if (join->session->killed)
2677
2606
join->session->send_kill_message();
2678
return NESTED_LOOP_KILLED;
2607
return NESTED_LOOP_KILLED; // Aborted by user /* purecov: inspected */
2680
optimizer::SqlSelect *select= join_tab->select;
2609
SQL_SELECT *select=join_tab->select;
2681
2610
if (rc == NESTED_LOOP_OK &&
2682
2611
(!join_tab->cache.select || !join_tab->cache.select->skip_record()))
2708
2637
join_tab->readCachedRecord(); // Restore current record
2709
2638
reset_cache_write(&join_tab->cache);
2710
2639
if (error > 0) // Fatal error
2711
return NESTED_LOOP_ERROR;
2640
return NESTED_LOOP_ERROR; /* purecov: inspected */
2712
2641
for (JoinTable *tmp2=join->join_tab; tmp2 != join_tab ; tmp2++)
2713
2642
tmp2->table->status=tmp2->status;
2714
2643
return NESTED_LOOP_OK;
2749
2678
if (join->do_send_rows)
2750
2679
error=join->result->send_data(*join->fields);
2752
return NESTED_LOOP_ERROR;
2681
return NESTED_LOOP_ERROR; /* purecov: inspected */
2753
2682
if (++join->send_records >= join->unit->select_limit_cnt && join->do_send_rows)
2755
2684
if (join->select_options & OPTION_FOUND_ROWS)
2758
2687
if ((join->tables == 1) && !join->tmp_table && !join->sort_and_group
2759
2688
&& !join->send_group_parts && !join->having && !jt->select_cond &&
2760
2689
!(jt->select && jt->select->quick) &&
2761
(jt->table->cursor->getEngine()->check_flag(HTON_BIT_STATS_RECORDS_IS_EXACT)) &&
2690
(jt->table->file->ha_table_flags() & HA_STATS_RECORDS_IS_EXACT) &&
2762
2691
(jt->ref.key < 0))
2764
2693
/* Join over all rows in table; Return number of found rows */
2819
2748
join->found_records++;
2820
if ((error=table->cursor->ha_write_row(table->record[0])))
2749
if ((error=table->file->ha_write_row(table->record[0])))
2822
if (!table->cursor->is_fatal_error(error, HA_CHECK_DUP))
2751
if (!table->file->is_fatal_error(error, HA_CHECK_DUP))
2824
2753
if (create_myisam_from_heap(join->session, table,
2825
2754
join->tmp_table_param.start_recinfo,
2868
2797
if (item->maybe_null)
2869
2798
group->buff[-1]= (char) group->field->is_null();
2871
if (!table->cursor->index_read_map(table->record[1],
2800
if (!table->file->index_read_map(table->record[1],
2872
2801
join->tmp_table_param.group_buff,
2874
2803
HA_READ_KEY_EXACT))
2875
2804
{ /* Update old record */
2876
2805
table->restoreRecord();
2877
2806
update_tmptable_sum_func(join->sum_funcs,table);
2878
if ((error= table->cursor->ha_update_row(table->record[1],
2807
if ((error= table->file->ha_update_row(table->record[1],
2879
2808
table->record[0])))
2881
table->print_error(error,MYF(0));
2882
return NESTED_LOOP_ERROR;
2810
table->file->print_error(error,MYF(0)); /* purecov: inspected */
2811
return NESTED_LOOP_ERROR; /* purecov: inspected */
2884
2813
return NESTED_LOOP_OK;
2900
2829
init_tmptable_sum_functions(join->sum_funcs);
2901
2830
copy_funcs(join->tmp_table_param.items_to_copy);
2902
if ((error=table->cursor->ha_write_row(table->record[0])))
2831
if ((error=table->file->ha_write_row(table->record[0])))
2904
2833
if (create_myisam_from_heap(join->session, table,
2905
2834
join->tmp_table_param.start_recinfo,
2908
2837
return NESTED_LOOP_ERROR; // Not a table_is_full error
2909
2838
/* Change method to update rows */
2910
table->cursor->ha_index_init(0, 0);
2839
table->file->ha_index_init(0, 0);
2911
2840
join->join_tab[join->tables-1].next_select= end_unique_update;
2913
2842
join->send_records++;
2925
2854
if (join->session->killed) // Aborted by user
2927
2856
join->session->send_kill_message();
2928
return NESTED_LOOP_KILLED;
2857
return NESTED_LOOP_KILLED; /* purecov: inspected */
2931
2860
init_tmptable_sum_functions(join->sum_funcs);
2932
2861
copy_fields(&join->tmp_table_param); // Groups are copied twice.
2933
2862
copy_funcs(join->tmp_table_param.items_to_copy);
2935
if (!(error= table->cursor->ha_write_row(table->record[0])))
2864
if (!(error= table->file->ha_write_row(table->record[0])))
2936
2865
join->send_records++; // New group
2939
if ((int) table->get_dup_key(error) < 0)
2868
if ((int) table->file->get_dup_key(error) < 0)
2941
table->print_error(error,MYF(0));
2942
return NESTED_LOOP_ERROR;
2870
table->file->print_error(error,MYF(0)); /* purecov: inspected */
2871
return NESTED_LOOP_ERROR; /* purecov: inspected */
2944
if (table->cursor->rnd_pos(table->record[1],table->cursor->dup_ref))
2873
if (table->file->rnd_pos(table->record[1],table->file->dup_ref))
2946
table->print_error(error,MYF(0));
2947
return NESTED_LOOP_ERROR;
2875
table->file->print_error(error,MYF(0)); /* purecov: inspected */
2876
return NESTED_LOOP_ERROR; /* purecov: inspected */
2949
2878
table->restoreRecord();
2950
2879
update_tmptable_sum_func(join->sum_funcs,table);
2951
if ((error= table->cursor->ha_update_row(table->record[1],
2880
if ((error= table->file->ha_update_row(table->record[1],
2952
2881
table->record[0])))
2954
table->print_error(error,MYF(0));
2955
return NESTED_LOOP_ERROR;
2883
table->file->print_error(error,MYF(0)); /* purecov: inspected */
2884
return NESTED_LOOP_ERROR; /* purecov: inspected */
2958
2887
return NESTED_LOOP_OK;
3075
3004
for (; group ; group=group->next)
3077
Cached_item *tmp= new_Cached_item(join->session, *group->item);
3006
Cached_item *tmp=new_Cached_item(join->session, *group->item, false);
3078
3007
if (!tmp || join->group_fields.push_front(tmp))
3243
3172
/** Save const tables first as used tables. */
3244
static void set_position(JOIN *join,
3247
optimizer::KeyUse *key)
3173
static void set_position(JOIN *join,uint32_t idx,JoinTable *table,KeyUse *key)
3249
3175
optimizer::Position tmp_pos(1.0, /* This is a const table */
3299
3225
Apply heuristic: pre-sort all access plans with respect to the number of
3300
3226
records accessed.
3302
internal::my_qsort(join->best_ref + join->const_tables,
3303
join->tables - join->const_tables, sizeof(JoinTable*),
3304
straight_join ? join_tab_cmp_straight : join_tab_cmp);
3228
my_qsort(join->best_ref + join->const_tables,
3229
join->tables - join->const_tables, sizeof(JoinTable*),
3230
straight_join ? join_tab_cmp_straight : join_tab_cmp);
3305
3231
if (straight_join)
3307
3233
optimize_straight_join(join, join_tables);
3358
3284
double record_count,
3361
optimizer::KeyUse *best_key= NULL;
3362
uint32_t best_max_key_part= 0;
3287
KeyUse *best_key= 0;
3288
uint32_t best_max_key_part= 0;
3363
3289
bool found_constraint= 0;
3364
double best= DBL_MAX;
3365
double best_time= DBL_MAX;
3366
double records= DBL_MAX;
3290
double best= DBL_MAX;
3291
double best_time= DBL_MAX;
3292
double records= DBL_MAX;
3367
3293
table_map best_ref_depends_map= 0;
3372
3298
{ /* Use key if possible */
3373
3299
Table *table= s->table;
3374
optimizer::KeyUse *keyuse= NULL;
3375
optimizer::KeyUse *start_key= NULL;
3300
KeyUse *keyuse,*start_key=0;
3376
3301
double best_records= DBL_MAX;
3377
3302
uint32_t max_key_part=0;
3379
3304
/* Test how we can use keys */
3380
3305
rec= s->records/MATCHING_ROWS_IN_OTHER_TABLE; // Assumed records/key
3381
for (keyuse= s->keyuse; keyuse->getTable() == table; )
3306
for (keyuse=s->keyuse ; keyuse->table == table ;)
3383
3308
key_part_map found_part= 0;
3384
3309
table_map found_ref= 0;
3385
uint32_t key= keyuse->getKey();
3386
KEY *keyinfo= table->key_info + key;
3310
uint32_t key= keyuse->key;
3311
KEY *keyinfo= table->key_info+key;
3387
3312
/* Bitmap of keyparts where the ref access is over 'keypart=const': */
3388
3313
key_part_map const_part= 0;
3389
3314
/* The or-null keypart in ref-or-null access: */
3395
3320
do /* For each keypart */
3397
uint32_t keypart= keyuse->getKeypart();
3322
uint32_t keypart= keyuse->keypart;
3398
3323
table_map best_part_found_ref= 0;
3399
3324
double best_prev_record_reads= DBL_MAX;
3405
3330
if 1. expression doesn't refer to forward tables
3406
3331
2. we won't get two ref-or-null's
3408
if (! (remaining_tables & keyuse->getUsedTables()) &&
3409
! (ref_or_null_part && (keyuse->getOptimizeFlags() &
3410
KEY_OPTIMIZE_REF_OR_NULL)))
3333
if (!(remaining_tables & keyuse->used_tables) &&
3334
!(ref_or_null_part && (keyuse->optimize &
3335
KEY_OPTIMIZE_REF_OR_NULL)))
3412
found_part|= keyuse->getKeypartMap();
3413
if (! (keyuse->getUsedTables() & ~join->const_table_map))
3414
const_part|= keyuse->getKeypartMap();
3337
found_part|= keyuse->keypart_map;
3338
if (!(keyuse->used_tables & ~join->const_table_map))
3339
const_part|= keyuse->keypart_map;
3416
3341
double tmp2= prev_record_reads(join, idx, (found_ref |
3417
keyuse->getUsedTables()));
3342
keyuse->used_tables));
3418
3343
if (tmp2 < best_prev_record_reads)
3420
best_part_found_ref= keyuse->getUsedTables() & ~join->const_table_map;
3345
best_part_found_ref= keyuse->used_tables & ~join->const_table_map;
3421
3346
best_prev_record_reads= tmp2;
3423
if (rec > keyuse->getTableRows())
3424
rec= keyuse->getTableRows();
3348
if (rec > keyuse->ref_table_rows)
3349
rec= keyuse->ref_table_rows;
3426
3351
If there is one 'key_column IS NULL' expression, we can
3427
3352
use this ref_or_null optimisation of this field
3429
if (keyuse->getOptimizeFlags() & KEY_OPTIMIZE_REF_OR_NULL)
3430
ref_or_null_part|= keyuse->getKeypartMap();
3354
if (keyuse->optimize & KEY_OPTIMIZE_REF_OR_NULL)
3355
ref_or_null_part |= keyuse->keypart_map;
3434
} while (keyuse->getTable() == table && keyuse->getKey() == key &&
3435
keyuse->getKeypart() == keypart);
3436
found_ref|= best_part_found_ref;
3437
} while (keyuse->getTable() == table && keyuse->getKey() == key);
3359
} while (keyuse->table == table && keyuse->key == key &&
3360
keyuse->keypart == keypart);
3361
found_ref|= best_part_found_ref;
3362
} while (keyuse->table == table && keyuse->key == key);
3440
3365
Assume that that each key matches a proportional part of table.
3525
3450
if (table->covering_keys.test(key))
3527
3452
/* we can use only index tree */
3528
tmp= record_count * table->cursor->index_only_read_time(key, tmp);
3453
tmp= record_count * table->file->index_only_read_time(key, tmp);
3531
3456
tmp= record_count * min(tmp,s->worst_seeks);
3539
3464
Set tmp to (previous record count) * (records / combination)
3541
3466
if ((found_part & 1) &&
3542
(!(table->index_flags(key) & HA_ONLY_WHOLE_INDEX) ||
3543
found_part == PREV_BITS(uint, keyinfo->key_parts)))
3467
(!(table->file->index_flags(key, 0, 0) & HA_ONLY_WHOLE_INDEX) ||
3468
found_part == PREV_BITS(uint,keyinfo->key_parts)))
3545
3470
max_key_part= max_part_bit(found_part);
3690
3615
if (table->covering_keys.test(key))
3692
3617
/* we can use only index tree */
3693
tmp= record_count * table->cursor->index_only_read_time(key, tmp);
3618
tmp= record_count * table->file->index_only_read_time(key, tmp);
3696
3621
tmp= record_count * min(tmp,s->worst_seeks);
3744
3669
if ((records >= s->found_records || best > s->read_time) && // (1)
3745
! (s->quick && best_key && s->quick->index == best_key->getKey() && // (2)
3746
best_max_key_part >= s->table->quick_key_parts[best_key->getKey()]) &&// (2)
3747
! ((s->table->cursor->getEngine()->check_flag(HTON_BIT_TABLE_SCAN_ON_INDEX)) && // (3)
3748
! s->table->covering_keys.none() && best_key && !s->quick) && // (3)
3749
! (s->table->force_index && best_key && !s->quick)) // (4)
3670
!(s->quick && best_key && s->quick->index == best_key->key && // (2)
3671
best_max_key_part >= s->table->quick_key_parts[best_key->key]) &&// (2)
3672
!((s->table->file->ha_table_flags() & HA_TABLE_SCAN_ON_INDEX) && // (3)
3673
! s->table->covering_keys.none() && best_key && !s->quick) &&// (3)
3674
!(s->table->force_index && best_key && !s->quick)) // (4)
3750
3675
{ // Check full join
3751
3676
ha_rows rnd_records= s->found_records;
4004
3929
JoinTable *best_table; // the next plan node to be added to the curr QEP
4006
3931
/* number of tables that remain to be optimized */
4007
size_remain= internal::my_count_bits(remaining_tables);
3932
size_remain= my_count_bits(remaining_tables);
4010
3935
/* Find the extension of the current QEP with the lowest cost */
4497
static bool make_join_select(JOIN *join,
4498
optimizer::SqlSelect *select,
4422
static bool make_join_select(JOIN *join,SQL_SELECT *select,COND *cond)
4501
4424
Session *session= join->session;
4502
4425
optimizer::Position cur_pos;
4756
4679
current_map, 0)))
4758
tab->cache.select= (optimizer::SqlSelect*)
4759
session->memdup((unsigned char*) sel, sizeof(optimizer::SqlSelect));
4681
tab->cache.select= (SQL_SELECT*)
4682
session->memdup((unsigned char*) sel, sizeof(SQL_SELECT));
4760
4683
tab->cache.select->cond= tmp;
4761
4684
tab->cache.select->read_tables= join->const_table_map;
4884
4807
Table *table=tab->table;
4885
4808
bool using_join_cache;
4886
4809
tab->read_record.table= table;
4887
tab->read_record.cursor= table->cursor;
4810
tab->read_record.file=table->file;
4888
4811
tab->next_select=sub_select; /* normal select */
4890
4813
TODO: don't always instruct first table's ref/range access method to
5025
4948
table->covering_keys.test(tab->select->quick->index))
5027
4950
table->key_read=1;
5028
table->cursor->extra(HA_EXTRA_KEYREAD);
4951
table->file->extra(HA_EXTRA_KEYREAD);
5030
4953
else if (!table->covering_keys.none() &&
5031
4954
!(tab->select && tab->select->quick))
5037
4960
is always faster than using a secondary index".
5039
4962
if (table->s->primary_key != MAX_KEY &&
5040
table->cursor->primary_key_is_clustered())
4963
table->file->primary_key_is_clustered())
5041
4964
tab->index= table->s->primary_key;
5043
4966
tab->index= table->find_shortest_key(&table->covering_keys);
5193
5116
if (select_options & SELECT_DESCRIBE)
5195
optimizer::ExplainPlan planner(join,
5200
planner.printPlan();
5118
select_describe(join, false, false, false, info);
5204
5122
join->join_free();
5536
5454
reclength= entry->s->reclength-offset;
5538
5456
entry->free_io_cache(); // Safety
5539
entry->cursor->info(HA_STATUS_VARIABLE);
5457
entry->file->info(HA_STATUS_VARIABLE);
5540
5458
if (entry->s->db_type() == heap_engine ||
5541
5459
(!entry->s->blob_fields &&
5542
((ALIGN_SIZE(reclength) + HASH_OVERHEAD) * entry->cursor->stats.records <
5460
((ALIGN_SIZE(reclength) + HASH_OVERHEAD) * entry->file->stats.records <
5543
5461
session->variables.sortbuff_size)))
5544
5462
error= remove_dup_with_hash_index(join->session, entry,
5545
5463
field_count, first_field,
5598
uint32_t table_count;
5599
uint32_t const_count;
5601
table_map found_const_table_map;
5602
table_map all_table_map;
5603
table_map found_ref;
5607
Table **table_vector= NULL;
5608
JoinTable *stat= NULL;
5609
JoinTable *stat_end= NULL;
5611
JoinTable **stat_ref= NULL;
5612
optimizer::KeyUse *keyuse= NULL;
5613
optimizer::KeyUse *start_keyuse= NULL;
5614
table_map outer_join= 0;
5515
uint32_t i,table_count,const_count,key;
5516
table_map found_const_table_map, all_table_map, found_ref, refs;
5517
key_map const_ref, eq_part;
5518
Table **table_vector;
5519
JoinTable *stat,*stat_end,*s,**stat_ref;
5520
KeyUse *keyuse,*start_keyuse;
5521
table_map outer_join=0;
5615
5522
vector<optimizer::SargableParam> sargables;
5616
5523
JoinTable *stat_vector[MAX_TABLES+1];
5617
5524
optimizer::Position *partial_pos;
5619
table_count= join->tables;
5620
stat= (JoinTable*) join->session->calloc(sizeof(JoinTable)*table_count);
5621
stat_ref= (JoinTable**) join->session->alloc(sizeof(JoinTable*)*MAX_TABLES);
5622
table_vector= (Table**) join->session->alloc(sizeof(Table*)*(table_count*2));
5526
table_count=join->tables;
5527
stat=(JoinTable*) join->session->calloc(sizeof(JoinTable)*table_count);
5528
stat_ref=(JoinTable**) join->session->alloc(sizeof(JoinTable*)*MAX_TABLES);
5529
table_vector=(Table**) join->session->alloc(sizeof(Table*)*(table_count*2));
5623
5530
if (! stat || ! stat_ref || ! table_vector)
5531
return 1; // Eom /* purecov: inspected */
5626
5533
join->best_ref=stat_vector;
5641
5548
s->needed_reg.reset();
5642
5549
table_vector[i]=s->table=table=tables->table;
5643
5550
table->pos_in_table_list= tables;
5644
error= table->cursor->info(HA_STATUS_VARIABLE | HA_STATUS_NO_LOCK);
5551
error= table->file->info(HA_STATUS_VARIABLE | HA_STATUS_NO_LOCK);
5647
table->print_error(error, MYF(0));
5554
table->file->print_error(error, MYF(0));
5650
5557
table->quick_keys.reset();
5659
5566
s->dependent= tables->dep_tables;
5660
5567
s->key_dependent= 0;
5661
table->quick_condition_rows= table->cursor->stats.records;
5568
if (tables->schema_table)
5569
table->file->stats.records= 2;
5570
table->quick_condition_rows= table->file->stats.records;
5663
5572
s->on_expr_ref= &tables->on_expr;
5664
5573
if (*s->on_expr_ref)
5666
5575
/* s is the only inner table of an outer join */
5667
if (!table->cursor->stats.records && !embedding)
5576
if (!table->file->stats.records && !embedding)
5668
5577
{ // Empty table
5669
5578
s->dependent= 0; // Ignore LEFT JOIN depend.
5670
set_position(join, const_count++, s, (optimizer::KeyUse*) 0);
5579
set_position(join,const_count++,s,(KeyUse*) 0);
5673
5582
outer_join|= table->map;
5691
5600
while (embedding);
5694
if ((table->cursor->stats.records <= 1) && !s->dependent &&
5695
(table->cursor->getEngine()->check_flag(HTON_BIT_STATS_RECORDS_IS_EXACT)) &&
5603
if ((table->file->stats.records <= 1) && !s->dependent &&
5604
(table->file->ha_table_flags() & HA_STATS_RECORDS_IS_EXACT) &&
5696
5605
!join->no_const_tables)
5698
set_position(join, const_count++, s, (optimizer::KeyUse*) 0);
5607
set_position(join,const_count++,s,(KeyUse*) 0);
5701
5610
stat_vector[i]=0;
5796
5705
TODO. Apply single row substitution to null complemented inner tables
5797
5706
for nested outer join operations.
5799
while (keyuse->getTable() == table)
5708
while (keyuse->table == table)
5801
if (! (keyuse->getVal()->used_tables() & ~join->const_table_map) &&
5802
keyuse->getVal()->is_null() && keyuse->isNullRejected())
5710
if (!(keyuse->val->used_tables() & ~join->const_table_map) &&
5711
keyuse->val->is_null() && keyuse->null_rejecting)
5804
5713
s->type= AM_CONST;
5805
5714
table->mark_as_null_row();
5806
5715
found_const_table_map|= table->map;
5807
5716
join->const_table_map|= table->map;
5808
set_position(join, const_count++, s, (optimizer::KeyUse*) 0);
5717
set_position(join,const_count++,s,(KeyUse*) 0);
5809
5718
goto more_const_tables_found;
5817
5726
// All dep. must be constants
5818
5727
if (s->dependent & ~(found_const_table_map))
5820
if (table->cursor->stats.records <= 1L &&
5821
(table->cursor->getEngine()->check_flag(HTON_BIT_STATS_RECORDS_IS_EXACT)) &&
5729
if (table->file->stats.records <= 1L &&
5730
(table->file->ha_table_flags() & HA_STATS_RECORDS_IS_EXACT) &&
5822
5731
!table->pos_in_table_list->embedding)
5823
5732
{ // system table
5825
5734
s->type= AM_SYSTEM;
5826
5735
join->const_table_map|=table->map;
5827
set_position(join, const_count++, s, (optimizer::KeyUse*) 0);
5736
set_position(join,const_count++,s,(KeyUse*) 0);
5828
5737
partial_pos= join->getSpecificPosInPartialPlan(const_count - 1);
5829
5738
if ((tmp= join_read_const_table(s, partial_pos)))
5840
5749
if ((keyuse=s->keyuse))
5842
5751
s->type= AM_REF;
5843
while (keyuse->getTable() == table)
5752
while (keyuse->table == table)
5845
start_keyuse= keyuse;
5846
key= keyuse->getKey();
5754
start_keyuse=keyuse;
5847
5756
s->keys.set(key); // QQ: remove this ?
5851
5760
eq_part.reset();
5854
if (keyuse->getVal()->type() != Item::NULL_ITEM &&
5855
! keyuse->getOptimizeFlags())
5763
if (keyuse->val->type() != Item::NULL_ITEM && !keyuse->optimize)
5857
if (! ((~found_const_table_map) & keyuse->getUsedTables()))
5858
const_ref.set(keyuse->getKeypart());
5765
if (!((~found_const_table_map) & keyuse->used_tables))
5766
const_ref.set(keyuse->keypart);
5860
refs|= keyuse->getUsedTables();
5861
eq_part.set(keyuse->getKeypart());
5768
refs|=keyuse->used_tables;
5769
eq_part.set(keyuse->keypart);
5864
} while (keyuse->getTable() == table && keyuse->getKey() == key);
5772
} while (keyuse->table == table && keyuse->key == key);
5866
5774
if (is_keymap_prefix(eq_part, table->key_info[key].key_parts) &&
5867
! table->pos_in_table_list->embedding)
5775
!table->pos_in_table_list->embedding)
5869
5777
if ((table->key_info[key].flags & (HA_NOSAME)) == HA_NOSAME)
5874
5782
ref_changed = 1;
5875
5783
s->type= AM_CONST;
5876
5784
join->const_table_map|= table->map;
5877
set_position(join, const_count++, s, start_keyuse);
5785
set_position(join,const_count++,s,start_keyuse);
5878
5786
if (create_ref_for_key(join, s, start_keyuse, found_const_table_map))
5880
5788
partial_pos= join->getSpecificPosInPartialPlan(const_count - 1);
5933
5841
/* Approximate found rows and time to read them */
5934
s->found_records=s->records=s->table->cursor->stats.records;
5935
s->read_time=(ha_rows) s->table->cursor->scan_time();
5842
s->found_records=s->records=s->table->file->stats.records;
5843
s->read_time=(ha_rows) s->table->file->scan_time();
5938
5846
Set a max range of how many seeks we can expect when using keys
5954
5862
!s->table->pos_in_table_list->embedding)
5956
5864
ha_rows records;
5957
optimizer::SqlSelect *select= NULL;
5958
select= optimizer::make_select(s->table, found_const_table_map, found_const_table_map, *s->on_expr_ref ? *s->on_expr_ref : conds, 1, &error);
5866
select= make_select(s->table, found_const_table_map, found_const_table_map, *s->on_expr_ref ? *s->on_expr_ref : conds, 1, &error);
5961
5869
records= get_quick_record_count(join->session, select, s->table, &s->const_keys, join->row_limit);
5971
5879
caller to abort with a zero row result.
5973
5881
join->const_table_map|= s->table->map;
5974
set_position(join, const_count++, s, (optimizer::KeyUse*) 0);
5882
set_position(join,const_count++,s,(KeyUse*) 0);
5975
5883
s->type= AM_CONST;
5976
5884
if (*s->on_expr_ref)
6001
5909
if (join->const_tables != join->tables)
6003
5911
optimize_keyuse(join, keyuse_array);
6004
DRIZZLE_QUERY_OPT_CHOOSE_PLAN_START(join->session->query.c_str(), join->session->thread_id);
6005
bool res= choose_plan(join, all_table_map & ~join->const_table_map);
6006
DRIZZLE_QUERY_OPT_CHOOSE_PLAN_DONE(res ? 1 : 0);
5912
if (choose_plan(join, all_table_map & ~join->const_table_map))
6232
6137
error=(int) cond->add(join_tab->select->cond);
6233
6138
join_tab->select_cond=join_tab->select->cond=cond;
6235
else if ((join_tab->select= optimizer::make_select(join_tab->table, 0, 0, cond, 0,
6140
else if ((join_tab->select= make_select(join_tab->table, 0, 0, cond, 0,
6237
6142
join_tab->select_cond=cond;
6239
6144
return(error ? true : false);