Commit bb15b9e2 authored by Alexander Barkov's avatar Alexander Barkov

MDEV-7950 Item_func::type() takes 0.26% in OLTP RO

parent bac6bbab
...@@ -59,6 +59,9 @@ struct TABLE_LIST; ...@@ -59,6 +59,9 @@ struct TABLE_LIST;
void item_init(void); /* Init item functions */ void item_init(void); /* Init item functions */
class Item_field; class Item_field;
class user_var_entry; class user_var_entry;
class JOIN;
struct KEY_FIELD;
struct SARGABLE_PARAM;
static inline uint32 static inline uint32
...@@ -1137,6 +1140,13 @@ public: ...@@ -1137,6 +1140,13 @@ public:
} }
virtual COND *remove_eq_conds(THD *thd, Item::cond_result *cond_value, virtual COND *remove_eq_conds(THD *thd, Item::cond_result *cond_value,
bool top_level); bool top_level);
virtual void add_key_fields(JOIN *join, KEY_FIELD **key_fields,
uint *and_level,
table_map usable_tables,
SARGABLE_PARAM **sargables)
{
return;
}
/* /*
Checks whether the item is: Checks whether the item is:
- a simple equality (field=field_item or field=constant_item), or - a simple equality (field=field_item or field=constant_item), or
......
...@@ -122,6 +122,10 @@ public: ...@@ -122,6 +122,10 @@ public:
class Item_bool_func :public Item_int_func class Item_bool_func :public Item_int_func
{ {
protected:
void add_key_fields_optimize_op(JOIN *join, KEY_FIELD **key_fields,
uint *and_level, table_map usable_tables,
SARGABLE_PARAM **sargables, bool equal_func);
public: public:
Item_bool_func() :Item_int_func() {} Item_bool_func() :Item_int_func() {}
Item_bool_func(Item *a) :Item_int_func(a) {} Item_bool_func(Item *a) :Item_int_func(a) {}
...@@ -434,6 +438,13 @@ public: ...@@ -434,6 +438,13 @@ public:
return min_max_arg_item->field->can_optimize_group_min_max(this, return min_max_arg_item->field->can_optimize_group_min_max(this,
const_item); const_item);
} }
void add_key_fields(JOIN *join, KEY_FIELD **key_fields,
uint *and_level, table_map usable_tables,
SARGABLE_PARAM **sargables)
{
return add_key_fields_optimize_op(join, key_fields, and_level,
usable_tables, sargables, false);
}
}; };
/** /**
...@@ -512,6 +523,9 @@ public: ...@@ -512,6 +523,9 @@ public:
const char *func_name() const { return "trigcond"; }; const char *func_name() const { return "trigcond"; };
bool const_item() const { return FALSE; } bool const_item() const { return FALSE; }
bool *get_trig_var() { return trig_var; } bool *get_trig_var() { return trig_var; }
void add_key_fields(JOIN *join, KEY_FIELD **key_fields,
uint *and_level, table_map usable_tables,
SARGABLE_PARAM **sargables);
}; };
class Item_func_not_all :public Item_func_not class Item_func_not_all :public Item_func_not
...@@ -567,6 +581,13 @@ public: ...@@ -567,6 +581,13 @@ public:
COND *build_equal_items(THD *thd, COND_EQUAL *inherited, COND *build_equal_items(THD *thd, COND_EQUAL *inherited,
bool link_item_fields, bool link_item_fields,
COND_EQUAL **cond_equal_ref); COND_EQUAL **cond_equal_ref);
void add_key_fields(JOIN *join, KEY_FIELD **key_fields,
uint *and_level, table_map usable_tables,
SARGABLE_PARAM **sargables)
{
return add_key_fields_optimize_op(join, key_fields, and_level,
usable_tables, sargables, true);
}
bool check_equality(THD *thd, COND_EQUAL *cond, List<Item> *eq_list); bool check_equality(THD *thd, COND_EQUAL *cond, List<Item> *eq_list);
/* /*
- If this equality is created from the subquery's IN-equality: - If this equality is created from the subquery's IN-equality:
...@@ -591,6 +612,13 @@ public: ...@@ -591,6 +612,13 @@ public:
cond_result eq_cmp_result() const { return COND_TRUE; } cond_result eq_cmp_result() const { return COND_TRUE; }
const char *func_name() const { return "<=>"; } const char *func_name() const { return "<=>"; }
Item *neg_transformer(THD *thd) { return 0; } Item *neg_transformer(THD *thd) { return 0; }
void add_key_fields(JOIN *join, KEY_FIELD **key_fields,
uint *and_level, table_map usable_tables,
SARGABLE_PARAM **sargables)
{
return add_key_fields_optimize_op(join, key_fields, and_level,
usable_tables, sargables, true);
}
}; };
...@@ -656,6 +684,8 @@ public: ...@@ -656,6 +684,8 @@ public:
optimize_type select_optimize() const { return OPTIMIZE_KEY; } optimize_type select_optimize() const { return OPTIMIZE_KEY; }
const char *func_name() const { return "<>"; } const char *func_name() const { return "<>"; }
Item *negated_item(); Item *negated_item();
void add_key_fields(JOIN *join, KEY_FIELD **key_fields, uint *and_level,
table_map usable_tables, SARGABLE_PARAM **sargables);
}; };
...@@ -712,6 +742,9 @@ public: ...@@ -712,6 +742,9 @@ public:
bool eval_not_null_tables(uchar *opt_arg); bool eval_not_null_tables(uchar *opt_arg);
void fix_after_pullout(st_select_lex *new_parent, Item **ref); void fix_after_pullout(st_select_lex *new_parent, Item **ref);
bool count_sargable_conds(uchar *arg); bool count_sargable_conds(uchar *arg);
void add_key_fields(JOIN *join, KEY_FIELD **key_fields,
uint *and_level, table_map usable_tables,
SARGABLE_PARAM **sargables);
}; };
...@@ -1392,6 +1425,8 @@ public: ...@@ -1392,6 +1425,8 @@ public:
} }
optimize_type select_optimize() const optimize_type select_optimize() const
{ return OPTIMIZE_KEY; } { return OPTIMIZE_KEY; }
void add_key_fields(JOIN *join, KEY_FIELD **key_fields, uint *and_level,
table_map usable_tables, SARGABLE_PARAM **sargables);
virtual void print(String *str, enum_query_type query_type); virtual void print(String *str, enum_query_type query_type);
enum Functype functype() const { return IN_FUNC; } enum Functype functype() const { return IN_FUNC; }
const char *func_name() const { return " IN "; } const char *func_name() const { return " IN "; }
...@@ -1436,6 +1471,8 @@ class Item_func_null_predicate :public Item_bool_func ...@@ -1436,6 +1471,8 @@ class Item_func_null_predicate :public Item_bool_func
public: public:
Item_func_null_predicate(Item *a) :Item_bool_func(a) { sargable= true; } Item_func_null_predicate(Item *a) :Item_bool_func(a) { sargable= true; }
optimize_type select_optimize() const { return OPTIMIZE_NULL; } optimize_type select_optimize() const { return OPTIMIZE_NULL; }
void add_key_fields(JOIN *join, KEY_FIELD **key_fields, uint *and_level,
table_map usable_tables, SARGABLE_PARAM **sargables);
CHARSET_INFO *compare_collation() const CHARSET_INFO *compare_collation() const
{ return args[0]->collation.collation; } { return args[0]->collation.collation; }
void fix_length_and_dec() { decimals=0; max_length=1; maybe_null=0; } void fix_length_and_dec() { decimals=0; max_length=1; maybe_null=0; }
...@@ -1587,6 +1624,8 @@ public: ...@@ -1587,6 +1624,8 @@ public:
*/ */
return compare_collation() == &my_charset_bin ? COND_TRUE : COND_OK; return compare_collation() == &my_charset_bin ? COND_TRUE : COND_OK;
} }
void add_key_fields(JOIN *join, KEY_FIELD **key_fields, uint *and_level,
table_map usable_tables, SARGABLE_PARAM **sargables);
const char *func_name() const { return "like"; } const char *func_name() const { return "like"; }
bool fix_fields(THD *thd, Item **ref); bool fix_fields(THD *thd, Item **ref);
void cleanup(); void cleanup();
...@@ -1781,6 +1820,9 @@ public: ...@@ -1781,6 +1820,9 @@ public:
COND_EQUAL **cond_equal_ref); COND_EQUAL **cond_equal_ref);
COND *remove_eq_conds(THD *thd, Item::cond_result *cond_value, COND *remove_eq_conds(THD *thd, Item::cond_result *cond_value,
bool top_level); bool top_level);
void add_key_fields(JOIN *join, KEY_FIELD **key_fields,
uint *and_level, table_map usable_tables,
SARGABLE_PARAM **sargables);
virtual void print(String *str, enum_query_type query_type); virtual void print(String *str, enum_query_type query_type);
void split_sum_func(THD *thd, Item **ref_pointer_array, List<Item> &fields); void split_sum_func(THD *thd, Item **ref_pointer_array, List<Item> &fields);
friend int setup_conds(THD *thd, TABLE_LIST *tables, TABLE_LIST *leaves, friend int setup_conds(THD *thd, TABLE_LIST *tables, TABLE_LIST *leaves,
...@@ -1971,6 +2013,9 @@ public: ...@@ -1971,6 +2013,9 @@ public:
COND *build_equal_items(THD *thd, COND_EQUAL *inherited, COND *build_equal_items(THD *thd, COND_EQUAL *inherited,
bool link_item_fields, bool link_item_fields,
COND_EQUAL **cond_equal_ref); COND_EQUAL **cond_equal_ref);
void add_key_fields(JOIN *join, KEY_FIELD **key_fields,
uint *and_level, table_map usable_tables,
SARGABLE_PARAM **sargables);
bool walk(Item_processor processor, bool walk_subquery, uchar *arg); bool walk(Item_processor processor, bool walk_subquery, uchar *arg);
Item *transform(Item_transformer transformer, uchar *arg); Item *transform(Item_transformer transformer, uchar *arg);
virtual void print(String *str, enum_query_type query_type); virtual void print(String *str, enum_query_type query_type);
...@@ -2125,6 +2170,8 @@ public: ...@@ -2125,6 +2170,8 @@ public:
COND *build_equal_items(THD *thd, COND_EQUAL *inherited, COND *build_equal_items(THD *thd, COND_EQUAL *inherited,
bool link_item_fields, bool link_item_fields,
COND_EQUAL **cond_equal_ref); COND_EQUAL **cond_equal_ref);
void add_key_fields(JOIN *join, KEY_FIELD **key_fields, uint *and_level,
table_map usable_tables, SARGABLE_PARAM **sargables);
}; };
inline bool is_cond_and(Item *item) inline bool is_cond_and(Item *item)
......
...@@ -288,6 +288,13 @@ public: ...@@ -288,6 +288,13 @@ public:
enum Functype rev_functype() const { return spatial_rel; } enum Functype rev_functype() const { return spatial_rel; }
bool is_null() { (void) val_int(); return null_value; } bool is_null() { (void) val_int(); return null_value; }
optimize_type select_optimize() const { return OPTIMIZE_OP; } optimize_type select_optimize() const { return OPTIMIZE_OP; }
void add_key_fields(JOIN *join, KEY_FIELD **key_fields,
uint *and_level, table_map usable_tables,
SARGABLE_PARAM **sargables)
{
return add_key_fields_optimize_op(join, key_fields, and_level,
usable_tables, sargables, false);
}
}; };
...@@ -404,7 +411,6 @@ class Item_func_isempty: public Item_bool_func ...@@ -404,7 +411,6 @@ class Item_func_isempty: public Item_bool_func
public: public:
Item_func_isempty(Item *a): Item_bool_func(a) {} Item_func_isempty(Item *a): Item_bool_func(a) {}
longlong val_int(); longlong val_int();
optimize_type select_optimize() const { return OPTIMIZE_NONE; }
const char *func_name() const { return "st_isempty"; } const char *func_name() const { return "st_isempty"; }
void fix_length_and_dec() { maybe_null= 1; } void fix_length_and_dec() { maybe_null= 1; }
}; };
...@@ -418,7 +424,6 @@ class Item_func_issimple: public Item_bool_func ...@@ -418,7 +424,6 @@ class Item_func_issimple: public Item_bool_func
public: public:
Item_func_issimple(Item *a): Item_bool_func(a) {} Item_func_issimple(Item *a): Item_bool_func(a) {}
longlong val_int(); longlong val_int();
optimize_type select_optimize() const { return OPTIMIZE_NONE; }
const char *func_name() const { return "st_issimple"; } const char *func_name() const { return "st_issimple"; }
void fix_length_and_dec() { maybe_null= 1; } void fix_length_and_dec() { maybe_null= 1; }
}; };
...@@ -428,7 +433,6 @@ class Item_func_isclosed: public Item_bool_func ...@@ -428,7 +433,6 @@ class Item_func_isclosed: public Item_bool_func
public: public:
Item_func_isclosed(Item *a): Item_bool_func(a) {} Item_func_isclosed(Item *a): Item_bool_func(a) {}
longlong val_int(); longlong val_int();
optimize_type select_optimize() const { return OPTIMIZE_NONE; }
const char *func_name() const { return "st_isclosed"; } const char *func_name() const { return "st_isclosed"; }
void fix_length_and_dec() { maybe_null= 1; } void fix_length_and_dec() { maybe_null= 1; }
}; };
......
...@@ -75,7 +75,7 @@ static bool update_ref_and_keys(THD *thd, DYNAMIC_ARRAY *keyuse, ...@@ -75,7 +75,7 @@ static bool update_ref_and_keys(THD *thd, DYNAMIC_ARRAY *keyuse,
JOIN_TAB *join_tab, JOIN_TAB *join_tab,
uint tables, COND *conds, uint tables, COND *conds,
table_map table_map, SELECT_LEX *select_lex, table_map table_map, SELECT_LEX *select_lex,
st_sargable_param **sargables); SARGABLE_PARAM **sargables);
static bool sort_and_filter_keyuse(THD *thd, DYNAMIC_ARRAY *keyuse, static bool sort_and_filter_keyuse(THD *thd, DYNAMIC_ARRAY *keyuse,
bool skip_unprefixed_keyparts); bool skip_unprefixed_keyparts);
static int sort_keyuse(KEYUSE *a,KEYUSE *b); static int sort_keyuse(KEYUSE *a,KEYUSE *b);
...@@ -3381,12 +3381,12 @@ static ha_rows get_quick_record_count(THD *thd, SQL_SELECT *select, ...@@ -3381,12 +3381,12 @@ static ha_rows get_quick_record_count(THD *thd, SQL_SELECT *select,
We form a bitmap of indexes that can be used for sargable predicates. We form a bitmap of indexes that can be used for sargable predicates.
Only such indexes are involved in range analysis. Only such indexes are involved in range analysis.
*/ */
typedef struct st_sargable_param struct SARGABLE_PARAM
{ {
Field *field; /* field against which to check sargability */ Field *field; /* field against which to check sargability */
Item **arg_value; /* values of potential keys for lookups */ Item **arg_value; /* values of potential keys for lookups */
uint num_values; /* number of values in the above array */ uint num_values; /* number of values in the above array */
} SARGABLE_PARAM; };
/** /**
...@@ -4127,7 +4127,7 @@ error: ...@@ -4127,7 +4127,7 @@ error:
*****************************************************************************/ *****************************************************************************/
/// Used when finding key fields /// Used when finding key fields
typedef struct key_field_t { struct KEY_FIELD {
Field *field; Field *field;
Item_func *cond; Item_func *cond;
Item *val; ///< May be empty if diff constant Item *val; ///< May be empty if diff constant
...@@ -4141,7 +4141,8 @@ typedef struct key_field_t { ...@@ -4141,7 +4141,8 @@ typedef struct key_field_t {
bool null_rejecting; bool null_rejecting;
bool *cond_guard; /* See KEYUSE::cond_guard */ bool *cond_guard; /* See KEYUSE::cond_guard */
uint sj_pred_no; /* See KEYUSE::sj_pred_no */ uint sj_pred_no; /* See KEYUSE::sj_pred_no */
} KEY_FIELD; };
/** /**
Merge new key definitions to old ones, remove those not used in both. Merge new key definitions to old ones, remove those not used in both.
...@@ -4602,240 +4603,269 @@ is_local_field (Item *field) ...@@ -4602,240 +4603,269 @@ is_local_field (Item *field)
operation operation
*/ */
static void
add_key_fields(JOIN *join, KEY_FIELD **key_fields, uint *and_level, void
COND *cond, table_map usable_tables, Item_cond_and::add_key_fields(JOIN *join, KEY_FIELD **key_fields,
SARGABLE_PARAM **sargables) uint *and_level, table_map usable_tables,
SARGABLE_PARAM **sargables)
{ {
if (cond->type() == Item_func::COND_ITEM) List_iterator_fast<Item> li(*argument_list());
{ KEY_FIELD *org_key_fields= *key_fields;
List_iterator_fast<Item> li(*((Item_cond*) cond)->argument_list());
KEY_FIELD *org_key_fields= *key_fields;
if (((Item_cond*) cond)->functype() == Item_func::COND_AND_FUNC) Item *item;
{ while ((item=li++))
Item *item; item->add_key_fields(join, key_fields, and_level, usable_tables,
while ((item=li++)) sargables);
add_key_fields(join, key_fields, and_level, item, usable_tables, for (; org_key_fields != *key_fields ; org_key_fields++)
sargables); org_key_fields->level= *and_level;
for (; org_key_fields != *key_fields ; org_key_fields++) }
org_key_fields->level= *and_level;
}
else void
{ Item_cond::add_key_fields(JOIN *join, KEY_FIELD **key_fields,
(*and_level)++; uint *and_level, table_map usable_tables,
add_key_fields(join, key_fields, and_level, li++, usable_tables, SARGABLE_PARAM **sargables)
sargables); {
Item *item; List_iterator_fast<Item> li(*argument_list());
while ((item=li++)) KEY_FIELD *org_key_fields= *key_fields;
{
KEY_FIELD *start_key_fields= *key_fields; (*and_level)++;
(*and_level)++; (li++)->add_key_fields(join, key_fields, and_level, usable_tables,
add_key_fields(join, key_fields, and_level, item, usable_tables, sargables);
sargables); Item *item;
*key_fields=merge_key_fields(org_key_fields,start_key_fields, while ((item=li++))
*key_fields,++(*and_level)); {
} KEY_FIELD *start_key_fields= *key_fields;
} (*and_level)++;
return; item->add_key_fields(join, key_fields, and_level, usable_tables,
sargables);
*key_fields= merge_key_fields(org_key_fields,start_key_fields,
*key_fields, ++(*and_level));
} }
}
void
Item_func_trig_cond::add_key_fields(JOIN *join, KEY_FIELD **key_fields,
uint *and_level, table_map usable_tables,
SARGABLE_PARAM **sargables)
{
/* /*
Subquery optimization: Conditions that are pushed down into subqueries Subquery optimization: Conditions that are pushed down into subqueries
are wrapped into Item_func_trig_cond. We process the wrapped condition are wrapped into Item_func_trig_cond. We process the wrapped condition
but need to set cond_guard for KEYUSE elements generated from it. but need to set cond_guard for KEYUSE elements generated from it.
*/ */
if (!join->group_list && !join->order &&
join->unit->item &&
join->unit->item->substype() == Item_subselect::IN_SUBS &&
!join->unit->is_union())
{ {
if (cond->type() == Item::FUNC_ITEM && KEY_FIELD *save= *key_fields;
((Item_func*)cond)->functype() == Item_func::TRIG_COND_FUNC) args[0]->add_key_fields(join, key_fields, and_level, usable_tables,
sargables);
// Indicate that this ref access candidate is for subquery lookup:
for (; save != *key_fields; save++)
save->cond_guard= get_trig_var();
}
}
void
Item_func_between::add_key_fields(JOIN *join, KEY_FIELD **key_fields,
uint *and_level, table_map usable_tables,
SARGABLE_PARAM **sargables)
{
/*
Build list of possible keys for 'a BETWEEN low AND high'.
It is handled similar to the equivalent condition
'a >= low AND a <= high':
*/
Item_field *field_item;
bool equal_func= false;
uint num_values= 2;
bool binary_cmp= (args[0]->real_item()->type() == Item::FIELD_ITEM)
? ((Item_field*) args[0]->real_item())->field->binary()
: true;
/*
Additional optimization: If 'low = high':
Handle as if the condition was "t.key = low".
*/
if (!negated && args[1]->eq(args[2], binary_cmp))
{
equal_func= true;
num_values= 1;
}
/*
Append keys for 'field <cmp> value[]' if the
condition is of the form::
'<field> BETWEEN value[1] AND value[2]'
*/
if (is_local_field(args[0]))
{
field_item= (Item_field *) (args[0]->real_item());
add_key_equal_fields(join, key_fields, *and_level, this,
field_item, equal_func, &args[1],
num_values, usable_tables, sargables);
}
/*
Append keys for 'value[0] <cmp> field' if the
condition is of the form:
'value[0] BETWEEN field1 AND field2'
*/
for (uint i= 1; i <= num_values; i++)
{
if (is_local_field(args[i]))
{ {
Item *cond_arg= ((Item_func*)cond)->arguments()[0]; field_item= (Item_field *) (args[i]->real_item());
if (!join->group_list && !join->order && add_key_equal_fields(join, key_fields, *and_level, this,
join->unit->item && field_item, equal_func, args,
join->unit->item->substype() == Item_subselect::IN_SUBS && 1, usable_tables, sargables);
!join->unit->is_union())
{
KEY_FIELD *save= *key_fields;
add_key_fields(join, key_fields, and_level, cond_arg, usable_tables,
sargables);
// Indicate that this ref access candidate is for subquery lookup:
for (; save != *key_fields; save++)
save->cond_guard= ((Item_func_trig_cond*)cond)->get_trig_var();
}
return;
} }
} }
}
/* If item is of type 'field op field/constant' add it to key_fields */
if (cond->type() != Item::FUNC_ITEM) void
return; Item_func_in::add_key_fields(JOIN *join, KEY_FIELD **key_fields,
Item_func *cond_func= (Item_func*) cond; uint *and_level, table_map usable_tables,
switch (cond_func->select_optimize()) { SARGABLE_PARAM **sargables)
case Item_func::OPTIMIZE_NONE: {
break; if (is_local_field(args[0]) && !(used_tables() & OUTER_REF_TABLE_BIT))
case Item_func::OPTIMIZE_KEY: {
DBUG_ASSERT(arg_count != 2);
add_key_equal_fields(join, key_fields, *and_level, this,
(Item_field*) (args[0]->real_item()), false,
args + 1, arg_count - 1, usable_tables, sargables);
}
}
void
Item_func_ne::add_key_fields(JOIN *join, KEY_FIELD **key_fields,
uint *and_level, table_map usable_tables,
SARGABLE_PARAM **sargables)
{
if (is_local_field(args[0]) && !(used_tables() & OUTER_REF_TABLE_BIT))
{
Item **values= is_local_field(args[1]) ? args : args + 1;
add_key_equal_fields(join, key_fields, *and_level, this,
(Item_field*) (args[0]->real_item()), false,
values, 1, usable_tables, sargables);
}
}
void
Item_func_like::add_key_fields(JOIN *join, KEY_FIELD **key_fields,
uint *and_level, table_map usable_tables,
SARGABLE_PARAM **sargables)
{
if (is_local_field(args[0]) &&
Item_func_like::select_optimize() == OPTIMIZE_OP)
{ {
Item **values;
/* /*
Build list of possible keys for 'a BETWEEN low AND high'. SELECT * FROM t1 WHERE field LIKE const_pattern
It is handled similar to the equivalent condition const_pattern starts with a non-wildcard character
'a >= low AND a <= high':
*/ */
if (cond_func->functype() == Item_func::BETWEEN) add_key_equal_fields(join, key_fields, *and_level, this,
{ (Item_field*) args[0]->real_item(), false,
Item_field *field_item; args + 1, 1, usable_tables, sargables);
bool equal_func= FALSE; }
uint num_values= 2; }
values= cond_func->arguments();
bool binary_cmp= (values[0]->real_item()->type() == Item::FIELD_ITEM)
? ((Item_field*)values[0]->real_item())->field->binary()
: TRUE;
/* void
Additional optimization: If 'low = high': Item_bool_func::add_key_fields_optimize_op(JOIN *join, KEY_FIELD **key_fields,
Handle as if the condition was "t.key = low". uint *and_level,
*/ table_map usable_tables,
if (!((Item_func_between*)cond_func)->negated && SARGABLE_PARAM **sargables,
values[1]->eq(values[2], binary_cmp)) bool equal_func)
{ {
equal_func= TRUE; /* If item is of type 'field op field/constant' add it to key_fields */
num_values= 1; if (is_local_field(args[0]))
} {
add_key_equal_fields(join, key_fields, *and_level, this,
(Item_field*) args[0]->real_item(), equal_func,
args + 1, 1, usable_tables, sargables);
}
if (is_local_field(args[1]))
{
add_key_equal_fields(join, key_fields, *and_level, this,
(Item_field*) args[1]->real_item(), equal_func,
args, 1, usable_tables, sargables);
}
}
/*
Append keys for 'field <cmp> value[]' if the
condition is of the form::
'<field> BETWEEN value[1] AND value[2]'
*/
if (is_local_field(values[0]))
{
field_item= (Item_field *) (values[0]->real_item());
add_key_equal_fields(join, key_fields, *and_level, cond_func,
field_item, equal_func, &values[1],
num_values, usable_tables, sargables);
}
/*
Append keys for 'value[0] <cmp> field' if the
condition is of the form:
'value[0] BETWEEN field1 AND field2'
*/
for (uint i= 1; i <= num_values; i++)
{
if (is_local_field(values[i]))
{
field_item= (Item_field *) (values[i]->real_item());
add_key_equal_fields(join, key_fields, *and_level, cond_func,
field_item, equal_func, values,
1, usable_tables, sargables);
}
}
} // if ( ... Item_func::BETWEEN)
// IN, NE void
else if (is_local_field (cond_func->key_item()) && Item_func_null_predicate::add_key_fields(JOIN *join, KEY_FIELD **key_fields,
!(cond_func->used_tables() & OUTER_REF_TABLE_BIT)) uint *and_level,
{ table_map usable_tables,
values= cond_func->arguments()+1; SARGABLE_PARAM **sargables)
if (cond_func->functype() == Item_func::NE_FUNC && {
is_local_field (cond_func->arguments()[1])) /* column_name IS [NOT] NULL */
values--; if (is_local_field(args[0]) && !(used_tables() & OUTER_REF_TABLE_BIT))
DBUG_ASSERT(cond_func->functype() != Item_func::IN_FUNC ||
cond_func->argument_count() != 2);
add_key_equal_fields(join, key_fields, *and_level, cond_func,
(Item_field*) (cond_func->key_item()->real_item()),
0, values,
cond_func->argument_count()-1,
usable_tables, sargables);
}
break;
}
case Item_func::OPTIMIZE_OP:
{ {
bool equal_func=(cond_func->functype() == Item_func::EQ_FUNC || Item *tmp= new Item_null;
cond_func->functype() == Item_func::EQUAL_FUNC); if (unlikely(!tmp)) // Should never be true
return;
add_key_equal_fields(join, key_fields, *and_level, this,
(Item_field*) args[0]->real_item(),
functype() == Item_func::ISNULL_FUNC,
&tmp, 1, usable_tables, sargables);
}
}
if (is_local_field (cond_func->arguments()[0]))
{ void
add_key_equal_fields(join, key_fields, *and_level, cond_func, Item_equal::add_key_fields(JOIN *join, KEY_FIELD **key_fields,
(Item_field*) (cond_func->arguments()[0])-> uint *and_level, table_map usable_tables,
real_item(), SARGABLE_PARAM **sargables)
equal_func, {
cond_func->arguments()+1, 1, usable_tables, Item *const_item= get_const();
sargables); Item_equal_fields_iterator it(*this);
} if (const_item)
if (is_local_field (cond_func->arguments()[1]) && {
cond_func->functype() != Item_func::LIKE_FUNC)
/*
For each field field1 from item_equal consider the equality
field1=const_item as a condition allowing an index access of the table
with field1 by the keys value of field1.
*/
while (it++)
{ {
add_key_equal_fields(join, key_fields, *and_level, cond_func, Field *equal_field= it.get_curr_field();
(Item_field*) (cond_func->arguments()[1])-> add_key_field(join, key_fields, *and_level, this, equal_field,
real_item(), TRUE, &const_item, 1, usable_tables, sargables);
equal_func,
cond_func->arguments(),1,usable_tables,
sargables);
} }
break;
} }
case Item_func::OPTIMIZE_NULL: else
/* column_name IS [NOT] NULL */ {
if (is_local_field (cond_func->arguments()[0]) && /*
!(cond_func->used_tables() & OUTER_REF_TABLE_BIT)) Consider all pairs of different fields included into item_equal.
{ For each of them (field1, field1) consider the equality
Item *tmp=new Item_null; field1=field2 as a condition allowing an index access of the table
if (unlikely(!tmp)) // Should never be true with field1 by the keys value of field2.
return; */
add_key_equal_fields(join, key_fields, *and_level, cond_func, Item_equal_fields_iterator fi(*this);
(Item_field*) (cond_func->arguments()[0])-> while (fi++)
real_item(), {
cond_func->functype() == Item_func::ISNULL_FUNC, Field *field= fi.get_curr_field();
&tmp, 1, usable_tables, sargables); Item *item;
} while ((item= it++))
break;
case Item_func::OPTIMIZE_EQUAL:
Item_equal *item_equal= (Item_equal *) cond;
Item *const_item= item_equal->get_const();
Item_equal_fields_iterator it(*item_equal);
if (const_item)
{
/*
For each field field1 from item_equal consider the equality
field1=const_item as a condition allowing an index access of the table
with field1 by the keys value of field1.
*/
while (it++)
{ {
Field *equal_field= it.get_curr_field(); Field *equal_field= it.get_curr_field();
add_key_field(join, key_fields, *and_level, cond_func, equal_field, if (!field->eq(equal_field))
TRUE, &const_item, 1, usable_tables, sargables);
}
}
else
{
/*
Consider all pairs of different fields included into item_equal.
For each of them (field1, field1) consider the equality
field1=field2 as a condition allowing an index access of the table
with field1 by the keys value of field2.
*/
Item_equal_fields_iterator fi(*item_equal);
while (fi++)
{
Field *field= fi.get_curr_field();
Item *item;
while ((item= it++))
{ {
Field *equal_field= it.get_curr_field(); add_key_field(join, key_fields, *and_level, this, field,
if (!field->eq(equal_field)) TRUE, &item, 1, usable_tables,
{ sargables);
add_key_field(join, key_fields, *and_level, cond_func, field,
TRUE, &item, 1, usable_tables,
sargables);
}
} }
it.rewind();
} }
it.rewind();
} }
break;
} }
} }
...@@ -5120,8 +5150,8 @@ static void add_key_fields_for_nj(JOIN *join, TABLE_LIST *nested_join_table, ...@@ -5120,8 +5150,8 @@ static void add_key_fields_for_nj(JOIN *join, TABLE_LIST *nested_join_table,
tables |= table->table->map; tables |= table->table->map;
} }
if (nested_join_table->on_expr) if (nested_join_table->on_expr)
add_key_fields(join, end, and_level, nested_join_table->on_expr, tables, nested_join_table->on_expr->add_key_fields(join, end, and_level, tables,
sargables); sargables);
} }
...@@ -5244,8 +5274,8 @@ update_ref_and_keys(THD *thd, DYNAMIC_ARRAY *keyuse,JOIN_TAB *join_tab, ...@@ -5244,8 +5274,8 @@ update_ref_and_keys(THD *thd, DYNAMIC_ARRAY *keyuse,JOIN_TAB *join_tab,
if (cond) if (cond)
{ {
KEY_FIELD *saved_field= field; KEY_FIELD *saved_field= field;
add_key_fields(join_tab->join, &end, &and_level, cond, normal_tables, cond->add_key_fields(join_tab->join, &end, &and_level, normal_tables,
sargables); sargables);
for (; field != end ; field++) for (; field != end ; field++)
{ {
...@@ -5268,9 +5298,10 @@ update_ref_and_keys(THD *thd, DYNAMIC_ARRAY *keyuse,JOIN_TAB *join_tab, ...@@ -5268,9 +5298,10 @@ update_ref_and_keys(THD *thd, DYNAMIC_ARRAY *keyuse,JOIN_TAB *join_tab,
into account as well. into account as well.
*/ */
if (*join_tab[i].on_expr_ref) if (*join_tab[i].on_expr_ref)
add_key_fields(join_tab->join, &end, &and_level, (*join_tab[i].on_expr_ref)->add_key_fields(join_tab->join, &end,
*join_tab[i].on_expr_ref, &and_level,
join_tab[i].table->map, sargables); join_tab[i].table->map,
sargables);
} }
/* Process ON conditions for the nested joins */ /* Process ON conditions for the nested joins */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment