2000-07-31 21:29:14 +02:00
|
|
|
/* Copyright (C) 2000 MySQL AB & MySQL Finland AB & TCX DataKonsult AB
|
2001-12-06 14:10:51 +02:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
2001-12-06 14:10:51 +02:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
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.
|
2001-12-06 14:10:51 +02:00
|
|
|
|
2000-07-31 21:29:14 +02:00
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
|
|
|
|
|
|
|
|
|
|
|
|
/* classes to use when handling where clause */
|
|
|
|
|
|
|
|
#ifdef __GNUC__
|
|
|
|
#pragma interface /* gcc class implementation */
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "procedure.h"
|
|
|
|
#include <myisam.h>
|
|
|
|
|
|
|
|
typedef struct keyuse_t {
|
|
|
|
TABLE *table;
|
|
|
|
Item *val; /* or value if no field */
|
|
|
|
uint key,keypart;
|
|
|
|
table_map used_tables;
|
|
|
|
} KEYUSE;
|
|
|
|
|
|
|
|
class store_key;
|
|
|
|
|
|
|
|
typedef struct st_table_ref
|
|
|
|
{
|
|
|
|
bool key_err;
|
|
|
|
uint key_parts; // num of ...
|
|
|
|
uint key_length; // length of key_buff
|
|
|
|
int key; // key no
|
|
|
|
byte *key_buff; // value to look for with key
|
|
|
|
byte *key_buff2; // key_buff+key_length
|
|
|
|
store_key **key_copy; //
|
|
|
|
Item **items; // val()'s for each keypart
|
|
|
|
table_map depend_map; // Table depends on these tables.
|
|
|
|
} TABLE_REF;
|
|
|
|
|
|
|
|
/*
|
|
|
|
** CACHE_FIELD and JOIN_CACHE is used on full join to cache records in outer
|
|
|
|
** table
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct st_cache_field {
|
|
|
|
char *str;
|
|
|
|
uint length,blob_length;
|
|
|
|
Field_blob *blob_field;
|
|
|
|
bool strip;
|
|
|
|
} CACHE_FIELD;
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct st_join_cache {
|
|
|
|
uchar *buff,*pos,*end;
|
|
|
|
uint records,record_nr,ptr_record,fields,length,blobs;
|
|
|
|
CACHE_FIELD *field,**blob_ptr;
|
|
|
|
SQL_SELECT *select;
|
|
|
|
} JOIN_CACHE;
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
** The structs which holds the join connections and join states
|
|
|
|
*/
|
|
|
|
|
|
|
|
enum join_type { JT_UNKNOWN,JT_SYSTEM,JT_CONST,JT_EQ_REF,JT_REF,JT_MAYBE_REF,
|
|
|
|
JT_ALL, JT_RANGE, JT_NEXT, JT_FT};
|
|
|
|
|
|
|
|
class JOIN;
|
|
|
|
|
|
|
|
typedef struct st_join_table {
|
|
|
|
TABLE *table;
|
2000-10-06 21:15:03 +03:00
|
|
|
KEYUSE *keyuse; /* pointer to first used key */
|
|
|
|
SQL_SELECT *select;
|
|
|
|
COND *select_cond;
|
|
|
|
QUICK_SELECT *quick;
|
|
|
|
Item *on_expr;
|
|
|
|
const char *info;
|
2000-07-31 21:29:14 +02:00
|
|
|
int (*read_first_record)(struct st_join_table *tab);
|
|
|
|
int (*next_select)(JOIN *,struct st_join_table *,bool);
|
|
|
|
READ_RECORD read_record;
|
2000-10-06 21:15:03 +03:00
|
|
|
double worst_seeks;
|
2000-07-31 21:29:14 +02:00
|
|
|
key_map const_keys; /* Keys with constant part */
|
|
|
|
key_map checked_keys; /* Keys checked in find_best */
|
|
|
|
key_map needed_reg;
|
|
|
|
ha_rows records,found_records,read_time;
|
|
|
|
table_map dependent,key_dependent;
|
2000-10-06 21:15:03 +03:00
|
|
|
uint keys; /* all keys with can be used */
|
2000-07-31 21:29:14 +02:00
|
|
|
uint use_quick,index;
|
|
|
|
uint status; // Save status for cache
|
|
|
|
uint used_fields,used_fieldlength,used_blobs;
|
2000-10-06 21:15:03 +03:00
|
|
|
enum join_type type;
|
|
|
|
bool cached_eq_ref_table,eq_ref_table,not_used_in_distinct;
|
2000-07-31 21:29:14 +02:00
|
|
|
TABLE_REF ref;
|
2000-10-06 21:15:03 +03:00
|
|
|
JOIN_CACHE cache;
|
2001-04-25 22:44:27 +03:00
|
|
|
JOIN *join;
|
2000-07-31 21:29:14 +02:00
|
|
|
} JOIN_TAB;
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct st_position { /* Used in find_best */
|
2000-10-06 21:15:03 +03:00
|
|
|
double records_read;
|
2000-07-31 21:29:14 +02:00
|
|
|
JOIN_TAB *table;
|
|
|
|
KEYUSE *key;
|
|
|
|
} POSITION;
|
|
|
|
|
|
|
|
|
2002-10-02 13:33:08 +03:00
|
|
|
class JOIN :public Sql_alloc
|
|
|
|
{
|
2000-07-31 21:29:14 +02:00
|
|
|
public:
|
|
|
|
JOIN_TAB *join_tab,**best_ref,**map2table;
|
|
|
|
TABLE **table,**all_tables,*sort_by_table;
|
|
|
|
uint tables,const_tables;
|
|
|
|
uint send_group_parts;
|
|
|
|
bool sort_and_group,first_record,full_join,group, no_field_update;
|
2001-04-12 00:54:35 +03:00
|
|
|
bool do_send_rows;
|
2002-01-23 02:52:26 +02:00
|
|
|
table_map const_table_map,found_const_table_map,outer_join;
|
2002-12-06 21:11:27 +02:00
|
|
|
ha_rows send_records,found_records,examined_rows,row_limit, select_limit;
|
2000-07-31 21:29:14 +02:00
|
|
|
POSITION positions[MAX_TABLES+1],best_positions[MAX_TABLES+1];
|
|
|
|
double best_read;
|
|
|
|
List<Item> *fields;
|
|
|
|
List<Item_buff> group_fields;
|
|
|
|
TABLE *tmp_table;
|
2003-01-25 02:25:52 +02:00
|
|
|
// used to store 2 possible tmp table of SELECT
|
|
|
|
TABLE *exec_tmp_table1, *exec_tmp_table2;
|
2000-07-31 21:29:14 +02:00
|
|
|
THD *thd;
|
|
|
|
Item_sum **sum_funcs;
|
|
|
|
Procedure *procedure;
|
|
|
|
Item *having;
|
2003-03-14 20:22:06 +01:00
|
|
|
Item *tmp_having; // To store Having when processed temporary table
|
2000-07-31 21:29:14 +02:00
|
|
|
uint select_options;
|
|
|
|
select_result *result;
|
|
|
|
TMP_TABLE_PARAM tmp_table_param;
|
|
|
|
MYSQL_LOCK *lock;
|
2002-05-08 23:14:40 +03:00
|
|
|
// unit structure (with global parameters) for this select
|
|
|
|
SELECT_LEX_UNIT *unit;
|
2002-05-12 23:46:42 +03:00
|
|
|
// select that processed
|
|
|
|
SELECT_LEX *select_lex;
|
2003-01-25 02:25:52 +02:00
|
|
|
|
|
|
|
JOIN *tmp_join; // copy of this JOIN to be used with temporary tables
|
2002-05-12 23:46:42 +03:00
|
|
|
|
|
|
|
bool select_distinct, //Is select distinct?
|
|
|
|
no_order, simple_order, simple_group,
|
|
|
|
skip_sort_order, need_tmp,
|
|
|
|
hidden_group_fields,
|
|
|
|
buffer_result;
|
|
|
|
DYNAMIC_ARRAY keyuse;
|
|
|
|
Item::cond_result cond_value;
|
2003-01-25 02:25:52 +02:00
|
|
|
List<Item> all_fields; // to store all fields that used in query
|
|
|
|
//Above list changed to use temporary table
|
|
|
|
List<Item> tmp_all_fields1, tmp_all_fields2, tmp_all_fields3;
|
|
|
|
//Part, shared with list above, emulate following list
|
|
|
|
List<Item> tmp_fields_list1, tmp_fields_list2, tmp_fields_list3;
|
2002-05-12 23:46:42 +03:00
|
|
|
List<Item> & fields_list; // hold field list passed to mysql_select
|
|
|
|
int error;
|
|
|
|
|
|
|
|
ORDER *order, *group_list, *proc_param; //hold parameters of mysql_select
|
|
|
|
COND *conds; // ---"---
|
|
|
|
TABLE_LIST *tables_list; //hold 'tables' parameter of mysql_selec
|
|
|
|
SQL_SELECT *select; //created in optimisation phase
|
2003-01-25 02:25:52 +02:00
|
|
|
Item **ref_pointer_array; //used pointer reference for this select
|
|
|
|
// Copy of above to be used with different lists
|
|
|
|
Item **items0, **items1, **items2, **items3;
|
|
|
|
uint ref_pointer_array_size; // size of above in bytes
|
2002-05-12 23:46:42 +03:00
|
|
|
const char *zero_result_cause; // not 0 if exec must return zero result
|
2002-05-26 22:50:32 +03:00
|
|
|
|
2003-01-25 02:25:52 +02:00
|
|
|
bool union_part; // this subselect is part of union
|
|
|
|
bool optimized; // flag to avoid double optimization in EXPLAIN
|
2002-05-12 23:46:42 +03:00
|
|
|
|
|
|
|
JOIN(THD *thd, List<Item> &fields,
|
|
|
|
ulong select_options, select_result *result):
|
|
|
|
join_tab(0),
|
|
|
|
table(0),
|
|
|
|
tables(0), const_tables(0),
|
|
|
|
sort_and_group(0), first_record(0),
|
|
|
|
do_send_rows(1),
|
|
|
|
send_records(0), found_records(0), examined_rows(0),
|
2003-01-25 02:25:52 +02:00
|
|
|
exec_tmp_table1(0), exec_tmp_table2(0),
|
2002-05-12 23:46:42 +03:00
|
|
|
thd(thd),
|
|
|
|
sum_funcs(0),
|
2002-09-03 09:50:36 +03:00
|
|
|
procedure(0),
|
2003-01-25 02:25:52 +02:00
|
|
|
having(0), tmp_having(0),
|
2002-05-12 23:46:42 +03:00
|
|
|
select_options(select_options),
|
|
|
|
result(result),
|
|
|
|
lock(thd->lock),
|
|
|
|
select_lex(0), //for safety
|
2003-01-25 02:25:52 +02:00
|
|
|
tmp_join(0),
|
2002-05-12 23:46:42 +03:00
|
|
|
select_distinct(test(select_options & SELECT_DISTINCT)),
|
|
|
|
no_order(0), simple_order(0), simple_group(0), skip_sort_order(0),
|
|
|
|
need_tmp(0),
|
|
|
|
hidden_group_fields (0), /*safety*/
|
|
|
|
buffer_result(test(select_options & OPTION_BUFFER_RESULT) &&
|
|
|
|
!test(select_options & OPTION_FOUND_ROWS)),
|
|
|
|
all_fields(fields),
|
|
|
|
fields_list(fields),
|
2002-10-02 13:33:08 +03:00
|
|
|
error(0),
|
2002-05-12 23:46:42 +03:00
|
|
|
select(0),
|
2003-01-25 02:25:52 +02:00
|
|
|
ref_pointer_array(0), items0(0), items1(0), items2(0), items3(0),
|
|
|
|
ref_pointer_array_size(0),
|
|
|
|
zero_result_cause(0),
|
|
|
|
optimized(0)
|
2002-05-12 23:46:42 +03:00
|
|
|
{
|
|
|
|
fields_list = fields;
|
|
|
|
bzero((char*) &keyuse,sizeof(keyuse));
|
|
|
|
tmp_table_param.copy_field=0;
|
|
|
|
tmp_table_param.end_write_records= HA_POS_ERROR;
|
|
|
|
}
|
|
|
|
|
2003-01-25 02:25:52 +02:00
|
|
|
int prepare(Item ***rref_pointer_array, TABLE_LIST *tables, uint wind_num,
|
|
|
|
COND *conds, uint og_num, ORDER *order, ORDER *group,
|
|
|
|
Item *having, ORDER *proc_param, SELECT_LEX *select,
|
2003-03-06 17:02:10 +02:00
|
|
|
SELECT_LEX_UNIT *unit, bool tables_and_fields_initied);
|
2002-05-12 23:46:42 +03:00
|
|
|
int optimize();
|
2002-05-26 22:50:32 +03:00
|
|
|
int reinit();
|
2002-05-12 23:46:42 +03:00
|
|
|
void exec();
|
2002-11-14 00:26:18 +02:00
|
|
|
int cleanup(THD *thd);
|
2003-01-25 02:25:52 +02:00
|
|
|
void restore_tmp();
|
|
|
|
|
|
|
|
inline void init_items_ref_array()
|
|
|
|
{
|
|
|
|
items0= ref_pointer_array + all_fields.elements;
|
|
|
|
ref_pointer_array_size= all_fields.elements*sizeof(Item*);
|
|
|
|
memcpy(items0, ref_pointer_array, ref_pointer_array_size);
|
|
|
|
}
|
2000-07-31 21:29:14 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct st_select_check {
|
|
|
|
uint const_ref,reg_ref;
|
|
|
|
} SELECT_CHECK;
|
|
|
|
|
|
|
|
extern const char *join_type_str[];
|
|
|
|
void TEST_join(JOIN *join);
|
|
|
|
|
|
|
|
/* Extern functions in sql_select.cc */
|
|
|
|
bool store_val_in_field(Field *field,Item *val);
|
|
|
|
TABLE *create_tmp_table(THD *thd,TMP_TABLE_PARAM *param,List<Item> &fields,
|
|
|
|
ORDER *group, bool distinct, bool save_sum_fields,
|
2002-12-06 21:11:27 +02:00
|
|
|
ulong select_options, ha_rows rows_limit);
|
2000-07-31 21:29:14 +02:00
|
|
|
void free_tmp_table(THD *thd, TABLE *entry);
|
2001-01-28 21:35:50 +02:00
|
|
|
void count_field_types(TMP_TABLE_PARAM *param, List<Item> &fields,
|
|
|
|
bool reset_with_sum_func);
|
2003-01-25 02:25:52 +02:00
|
|
|
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);
|
2000-07-31 21:29:14 +02:00
|
|
|
void copy_fields(TMP_TABLE_PARAM *param);
|
2003-01-28 08:38:28 +02:00
|
|
|
void copy_funcs(Item **func_ptr);
|
2002-12-06 21:11:27 +02:00
|
|
|
bool create_myisam_from_heap(THD *thd, TABLE *table, TMP_TABLE_PARAM *param,
|
|
|
|
int error, bool ignore_last_dupp_error);
|
2000-07-31 21:29:14 +02:00
|
|
|
|
|
|
|
/* functions from opt_sum.cc */
|
|
|
|
int opt_sum_query(TABLE_LIST *tables, List<Item> &all_fields,COND *conds);
|
|
|
|
|
|
|
|
|
|
|
|
/* class to copying an field/item to a key struct */
|
|
|
|
|
|
|
|
class store_key :public Sql_alloc
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
Field *to_field; // Store data here
|
|
|
|
char *null_ptr;
|
|
|
|
char err;
|
|
|
|
public:
|
2001-12-05 13:03:00 +02:00
|
|
|
store_key(THD *thd, Field *field_arg, char *ptr, char *null, uint length)
|
2000-07-31 21:29:14 +02:00
|
|
|
:null_ptr(null),err(0)
|
|
|
|
{
|
|
|
|
if (field_arg->type() == FIELD_TYPE_BLOB)
|
|
|
|
to_field=new Field_varstring(ptr, length, (uchar*) null, 1,
|
|
|
|
Field::NONE, field_arg->field_name,
|
2002-10-25 13:58:32 +05:00
|
|
|
field_arg->table, field_arg->charset());
|
2000-07-31 21:29:14 +02:00
|
|
|
else
|
|
|
|
{
|
2001-12-05 13:03:00 +02:00
|
|
|
to_field=field_arg->new_field(&thd->mem_root,field_arg->table);
|
2000-07-31 21:29:14 +02:00
|
|
|
if (to_field)
|
|
|
|
to_field->move_field(ptr, (uchar*) null, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
virtual ~store_key() {} /* Not actually needed */
|
|
|
|
virtual bool copy()=0;
|
|
|
|
virtual const char *name() const=0;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class store_key_field: public store_key
|
|
|
|
{
|
|
|
|
Copy_field copy_field;
|
|
|
|
const char *field_name;
|
|
|
|
public:
|
2001-12-05 13:03:00 +02:00
|
|
|
store_key_field(THD *thd, Field *to_field_arg, char *ptr, char *null_ptr_arg,
|
2000-07-31 21:29:14 +02:00
|
|
|
uint length, Field *from_field, const char *name_arg)
|
2001-12-05 13:03:00 +02:00
|
|
|
:store_key(thd, to_field_arg,ptr,
|
2000-07-31 21:29:14 +02:00
|
|
|
null_ptr_arg ? null_ptr_arg : from_field->maybe_null() ? &err
|
|
|
|
: NullS,length), field_name(name_arg)
|
|
|
|
{
|
|
|
|
if (to_field)
|
|
|
|
{
|
|
|
|
copy_field.set(to_field,from_field,0);
|
|
|
|
}
|
|
|
|
}
|
2002-12-03 13:08:25 +02:00
|
|
|
bool copy()
|
|
|
|
{
|
|
|
|
copy_field.do_copy(©_field);
|
|
|
|
return err != 0;
|
|
|
|
}
|
|
|
|
const char *name() const { return field_name; }
|
2000-07-31 21:29:14 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class store_key_item :public store_key
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
Item *item;
|
|
|
|
public:
|
2001-12-05 13:03:00 +02:00
|
|
|
store_key_item(THD *thd, Field *to_field_arg, char *ptr, char *null_ptr_arg,
|
2000-07-31 21:29:14 +02:00
|
|
|
uint length, Item *item_arg)
|
2001-12-05 13:03:00 +02:00
|
|
|
:store_key(thd, to_field_arg,ptr,
|
2000-07-31 21:29:14 +02:00
|
|
|
null_ptr_arg ? null_ptr_arg : item_arg->maybe_null ?
|
|
|
|
&err : NullS, length), item(item_arg)
|
|
|
|
{}
|
|
|
|
bool copy()
|
|
|
|
{
|
2002-12-03 13:08:25 +02:00
|
|
|
return item->save_in_field(to_field, 1) || err != 0;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
const char *name() const { return "func"; }
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class store_key_const_item :public store_key_item
|
|
|
|
{
|
|
|
|
bool inited;
|
|
|
|
public:
|
2001-12-05 13:03:00 +02:00
|
|
|
store_key_const_item(THD *thd, Field *to_field_arg, char *ptr,
|
2000-07-31 21:29:14 +02:00
|
|
|
char *null_ptr_arg, uint length,
|
|
|
|
Item *item_arg)
|
2001-12-05 13:03:00 +02:00
|
|
|
:store_key_item(thd, to_field_arg,ptr,
|
2000-07-31 21:29:14 +02:00
|
|
|
null_ptr_arg ? null_ptr_arg : item_arg->maybe_null ?
|
|
|
|
&err : NullS, length, item_arg), inited(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
bool copy()
|
|
|
|
{
|
|
|
|
if (!inited)
|
|
|
|
{
|
|
|
|
inited=1;
|
2002-12-03 13:08:25 +02:00
|
|
|
if (item->save_in_field(to_field, 1))
|
|
|
|
err= 1;
|
2000-07-31 21:29:14 +02:00
|
|
|
}
|
|
|
|
return err != 0;
|
|
|
|
}
|
|
|
|
const char *name() const { return "const"; }
|
|
|
|
};
|
|
|
|
|
|
|
|
bool cp_buffer_from_ref(TABLE_REF *ref);
|
2002-11-29 16:40:18 +02:00
|
|
|
bool error_if_full_join(JOIN *join);
|
2003-01-07 11:45:06 +02:00
|
|
|
void relink_tables(SELECT_LEX *select_lex);
|