1
/* -*- mode: c++; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
* vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
4
* Copyright (C) 2008 Sun Microsystems
6
* This program is free software; you can redistribute it and/or modify
7
* it under the terms of the GNU General Public License as published by
8
* the Free Software Foundation; version 2 of the License.
10
* This program is distributed in the hope that it will be useful,
11
* but WITHOUT ANY WARRANTY; without even the implied warranty of
12
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13
* GNU General Public License for more details.
15
* You should have received a copy of the GNU General Public License
16
* along with this program; if not, write to the Free Software
17
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
22
#include <drizzled/session.h>
23
#include <drizzled/table.h>
24
#include <drizzled/error.h>
25
#include <drizzled/join.h>
26
#include <drizzled/sql_base.h>
27
#include <drizzled/sql_select.h>
28
#include <drizzled/item/cmpfunc.h>
29
#include <drizzled/item/field.h>
30
#include <drizzled/item/outer_ref.h>
31
#include <drizzled/plugin/client.h>
37
Store the pointer to this item field into a list if not already there.
39
The method is used by Item::walk to collect all unique Item_field objects
40
from a tree of Items into a set of items represented as a list.
42
Item_cond::walk() and Item_func::walk() stop the evaluation of the
43
processor function for its arguments once the processor returns
44
true.Therefore in order to force this method being called for all item
45
arguments in a condition the method must return false.
47
@param arg pointer to a List<Item_field>
50
false to force the evaluation of collect_item_field_processor
51
for the subsequent items.
54
bool Item_field::collect_item_field_processor(unsigned char *arg)
56
List<Item_field> *item_list= (List<Item_field>*) arg;
57
List_iterator<Item_field> item_list_it(*item_list);
58
Item_field *curr_item;
59
while ((curr_item= item_list_it++))
61
if (curr_item->eq(this, 1))
62
return(false); /* Already in the set. */
64
item_list->push_back(this);
70
Check if an Item_field references some field from a list of fields.
72
Check whether the Item_field represented by 'this' references any
73
of the fields in the keyparts passed via 'arg'. Used with the
74
method Item::walk() to test whether any keypart in a sequence of
75
keyparts is referenced in an expression.
77
@param arg Field being compared, arg must be of type Field
80
true if 'this' references the field 'arg'
85
bool Item_field::find_item_in_field_list_processor(unsigned char *arg)
87
KeyPartInfo *first_non_group_part= *((KeyPartInfo **) arg);
88
KeyPartInfo *last_part= *(((KeyPartInfo **) arg) + 1);
89
KeyPartInfo *cur_part;
91
for (cur_part= first_non_group_part; cur_part != last_part; cur_part++)
93
if (field->eq(cur_part->field))
101
Mark field in read_map
104
This is used by filesort to register used fields in a a temporary
105
column read set or to register used fields in a view
108
bool Item_field::register_field_in_read_map(unsigned char *arg)
110
Table *table= (Table *) arg;
111
if (field->getTable() == table || !table)
112
field->getTable()->setReadSet(field->field_index);
118
Item_field::Item_field(Field *f)
119
:Item_ident(0, NULL, f->getTable()->getAlias(), f->field_name),
120
item_equal(0), no_const_subst(0),
121
have_privileges(0), any_privileges(0)
125
field_name and table_name should not point to garbage
126
if this item is to be reused
128
orig_table_name= orig_field_name= "";
133
Constructor used inside setup_wild().
135
Ensures that field, table, and database names will live as long as
136
Item_field (this is important in prepared statements).
139
Item_field::Item_field(Session *,
140
Name_resolution_context *context_arg,
142
Item_ident(context_arg,
143
f->getTable()->getShare()->getSchemaName(),
144
f->getTable()->getAlias(),
155
Item_field::Item_field(Name_resolution_context *context_arg,
156
const char *db_arg,const char *table_name_arg,
157
const char *field_name_arg) :
158
Item_ident(context_arg, db_arg,table_name_arg,field_name_arg),
166
Select_Lex *select= current_session->lex->current_select;
167
collation.set(DERIVATION_IMPLICIT);
169
if (select && select->parsing_place != IN_HAVING)
170
select->select_n_where_fields++;
174
Constructor need to process subselect with temporary tables (see Item)
177
Item_field::Item_field(Session *session, Item_field *item) :
178
Item_ident(session, item),
180
result_field(item->result_field),
181
item_equal(item->item_equal),
182
no_const_subst(item->no_const_subst),
183
have_privileges(item->have_privileges),
184
any_privileges(item->any_privileges)
186
collation.set(DERIVATION_IMPLICIT);
189
void Item_field::set_field(Field *field_par)
191
field=result_field=field_par; // for easy coding with fields
192
maybe_null=field->maybe_null();
193
decimals= field->decimals();
194
max_length= field_par->max_display_length();
195
table_name= field_par->getTable()->getAlias();
196
field_name= field_par->field_name;
197
db_name= field_par->getTable()->getShare()->getSchemaName();
198
alias_name_used= field_par->getTable()->alias_name_used;
199
unsigned_flag=test(field_par->flags & UNSIGNED_FLAG);
200
collation.set(field_par->charset(), field_par->derivation());
206
Reset this item to point to a field from the new temporary table.
207
This is used when we create a new temporary table for each execution
208
of prepared statement.
211
void Item_field::reset_field(Field *f)
214
/* 'name' is pointing at field->field_name of old field */
215
name= (char*) f->field_name;
219
String *Item_field::val_str(String *str)
222
if ((null_value=field->is_null()))
224
str->set_charset(str_value.charset());
225
return field->val_str(str,&str_value);
229
double Item_field::val_real()
232
if ((null_value=field->is_null()))
234
return field->val_real();
238
int64_t Item_field::val_int()
241
if ((null_value=field->is_null()))
243
return field->val_int();
247
my_decimal *Item_field::val_decimal(my_decimal *decimal_value)
249
if ((null_value= field->is_null()))
251
return field->val_decimal(decimal_value);
255
String *Item_field::str_result(String *str)
257
if ((null_value=result_field->is_null()))
259
str->set_charset(str_value.charset());
260
return result_field->val_str(str,&str_value);
263
bool Item_field::get_date(DRIZZLE_TIME *ltime,uint32_t fuzzydate)
265
if ((null_value=field->is_null()) || field->get_date(ltime,fuzzydate))
267
memset(ltime, 0, sizeof(*ltime));
273
bool Item_field::get_date_result(DRIZZLE_TIME *ltime,uint32_t fuzzydate)
275
if ((null_value=result_field->is_null()) ||
276
result_field->get_date(ltime,fuzzydate))
278
memset(ltime, 0, sizeof(*ltime));
284
bool Item_field::get_time(DRIZZLE_TIME *ltime)
286
if ((null_value=field->is_null()) || field->get_time(ltime))
288
memset(ltime, 0, sizeof(*ltime));
294
double Item_field::val_result()
296
if ((null_value=result_field->is_null()))
298
return result_field->val_real();
301
int64_t Item_field::val_int_result()
303
if ((null_value=result_field->is_null()))
305
return result_field->val_int();
309
my_decimal *Item_field::val_decimal_result(my_decimal *decimal_value)
311
if ((null_value= result_field->is_null()))
313
return result_field->val_decimal(decimal_value);
317
bool Item_field::val_bool_result()
319
if ((null_value= result_field->is_null()))
321
switch (result_field->result_type()) {
323
return result_field->val_int() != 0;
326
my_decimal decimal_value;
327
my_decimal *val= result_field->val_decimal(&decimal_value);
329
return !my_decimal_is_zero(val);
334
return result_field->val_real() != 0.0;
338
return 0; // Shut up compiler
343
bool Item_field::eq(const Item *item, bool) const
345
const Item *item_ptr= item->real_item();
346
if (item_ptr->type() != FIELD_ITEM)
349
const Item_field *item_field= static_cast<const Item_field *>(item_ptr);
350
if (item_field->field && field)
351
return item_field->field == field;
353
We may come here when we are trying to find a function in a GROUP BY
354
clause from the select list.
355
In this case the '100 % correct' way to do this would be to first
356
run fix_fields() on the GROUP BY item and then retry this function, but
357
I think it's better to relax the checking a bit as we will in
358
most cases do the correct thing by just checking the field name.
359
(In cases where we would choose wrong we would have to generate a
362
return (!my_strcasecmp(system_charset_info, item_field->name,
364
(!item_field->table_name || !table_name ||
365
(!my_strcasecmp(table_alias_charset, item_field->table_name,
367
(!item_field->db_name || !db_name ||
368
(item_field->db_name && !strcasecmp(item_field->db_name,
373
table_map Item_field::used_tables() const
375
if (field->getTable()->const_table)
376
return 0; // const item
377
return (depended_from ? OUTER_REF_TABLE_BIT : field->getTable()->map);
380
enum Item_result Item_field::result_type () const
382
return field->result_type();
386
Item_result Item_field::cast_to_int_type() const
388
return field->cast_to_int_type();
392
enum_field_types Item_field::field_type() const
394
return field->type();
398
void Item_field::fix_after_pullout(Select_Lex *new_parent, Item **)
400
if (new_parent == depended_from)
402
Name_resolution_context *ctx= new Name_resolution_context();
403
ctx->outer_context= NULL; // We don't build a complete name resolver
404
ctx->select_lex= new_parent;
405
ctx->first_name_resolution_table= context->first_name_resolution_table;
406
ctx->last_name_resolution_table= context->last_name_resolution_table;
411
bool Item_field::is_null()
413
return field->is_null();
417
Item *Item_field::get_tmp_table_item(Session *session)
419
Item_field *new_item= new Item_field(session, this);
421
new_item->field= new_item->result_field;
425
int64_t Item_field::val_int_endpoint(bool, bool *)
427
int64_t res= val_int();
428
return null_value? INT64_MIN : res;
433
Resolve the name of an outer select column reference.
435
The method resolves the column reference represented by 'this' as a column
436
present in outer selects that contain current select.
438
In prepared statements, because of cache, find_field_in_tables()
439
can resolve fields even if they don't belong to current context.
440
In this case this method only finds appropriate context and marks
441
current select as dependent. The found reference of field should be
442
provided in 'from_field'.
444
@param[in] session current thread
445
@param[in,out] from_field found field reference or (Field*)not_found_field
446
@param[in,out] reference view column if this item was resolved to a
450
This is the inner loop of Item_field::fix_fields:
452
for each outer query Q_k beginning from the inner-most one
454
search for a column or derived column named col_ref_i
455
[in table T_j] in the FROM clause of Q_k;
457
if such a column is not found
458
Search for a column or derived column named col_ref_i
459
[in table T_j] in the SELECT and GROUP clauses of Q_k.
464
1 column succefully resolved and fix_fields() should continue.
466
0 column fully fixed and fix_fields() should return false
472
Item_field::fix_outer_field(Session *session, Field **from_field, Item **reference)
474
enum_parsing_place place= NO_MATTER;
475
bool field_found= (*from_field != not_found_field);
476
bool upward_lookup= false;
479
If there are outer contexts (outer selects, but current select is
480
not derived table or view) try to resolve this reference in the
483
We treat each subselect as a separate namespace, so that different
484
subselects may contain columns with the same names. The subselects
485
are searched starting from the innermost.
487
Name_resolution_context *last_checked_context= context;
488
Item **ref= (Item **) not_found_item;
489
Select_Lex *current_sel= (Select_Lex *) session->lex->current_select;
490
Name_resolution_context *outer_context= 0;
491
Select_Lex *select= 0;
492
/* Currently derived tables cannot be correlated */
493
if (current_sel->master_unit()->first_select()->linkage !=
495
outer_context= context->outer_context;
498
outer_context= outer_context->outer_context)
500
select= outer_context->select_lex;
501
Item_subselect *prev_subselect_item=
502
last_checked_context->select_lex->master_unit()->item;
503
last_checked_context= outer_context;
506
place= prev_subselect_item->parsing_place;
508
If outer_field is set, field was already found by first call
509
to find_field_in_tables(). Only need to find appropriate context.
511
if (field_found && outer_context->select_lex !=
512
cached_table->select_lex)
515
In case of a view, find_field_in_tables() writes the pointer to
516
the found view field into '*reference', in other words, it
517
substitutes this Item_field with the found expression.
519
if (field_found || (*from_field= find_field_in_tables(session, this,
521
first_name_resolution_table,
523
last_name_resolution_table,
525
IGNORE_EXCEPT_NON_UNIQUE,
531
if (*from_field != view_ref_found)
533
prev_subselect_item->used_tables_cache|= (*from_field)->getTable()->map;
534
prev_subselect_item->const_item_cache= 0;
535
set_field(*from_field);
536
if (!last_checked_context->select_lex->having_fix_field &&
537
select->group_list.elements &&
538
(place == SELECT_LIST || place == IN_HAVING))
542
If an outer field is resolved in a grouping select then it
543
is replaced for an Item_outer_ref object. Otherwise an
544
Item_field object is used.
545
The new Item_outer_ref object is saved in the inner_refs_list of
546
the outer select. Here it is only created. It can be fixed only
547
after the original field has been fixed and this is done in the
548
fix_inner_refs() function.
551
if (!(rf= new Item_outer_ref(context, this)))
553
session->change_item_tree(reference, rf);
554
select->inner_refs_list.push_back(rf);
555
rf->in_sum_func= session->lex->in_sum_func;
558
A reference is resolved to a nest level that's outer or the same as
559
the nest level of the enclosing set function : adjust the value of
560
max_arg_level for the function if it's needed.
562
if (session->lex->in_sum_func &&
563
session->lex->in_sum_func->nest_level >= select->nest_level)
565
Item::Type ref_type= (*reference)->type();
566
set_if_bigger(session->lex->in_sum_func->max_arg_level,
568
set_field(*from_field);
570
mark_as_dependent(session, last_checked_context->select_lex,
571
context->select_lex, this,
572
((ref_type == REF_ITEM ||
573
ref_type == FIELD_ITEM) ?
574
(Item_ident*) (*reference) : 0));
580
Item::Type ref_type= (*reference)->type();
581
prev_subselect_item->used_tables_cache|=
582
(*reference)->used_tables();
583
prev_subselect_item->const_item_cache&=
584
(*reference)->const_item();
585
mark_as_dependent(session, last_checked_context->select_lex,
586
context->select_lex, this,
587
((ref_type == REF_ITEM || ref_type == FIELD_ITEM) ?
588
(Item_ident*) (*reference) :
591
A reference to a view field had been found and we
592
substituted it instead of this Item (find_field_in_tables
593
does it by assigning the new value to *reference), so now
594
we can return from this function.
602
/* Search in SELECT and GROUP lists of the outer select. */
603
if (place != IN_WHERE && place != IN_ON)
605
if (!(ref= resolve_ref_in_select_and_group(session, this, select)))
606
return -1; /* Some error occurred (e.g. ambiguous names). */
607
if (ref != not_found_item)
609
assert(*ref && (*ref)->fixed);
610
prev_subselect_item->used_tables_cache|= (*ref)->used_tables();
611
prev_subselect_item->const_item_cache&= (*ref)->const_item();
617
Reference is not found in this select => this subquery depend on
618
outer select (or we just trying to find wrong identifier, in this
619
case it does not matter which used tables bits we set)
621
prev_subselect_item->used_tables_cache|= OUTER_REF_TABLE_BIT;
622
prev_subselect_item->const_item_cache= 0;
628
if (ref == not_found_item && *from_field == not_found_field)
632
// We can't say exactly what absent table or field
633
my_error(ER_BAD_FIELD_ERROR, MYF(0), full_name(), session->where);
637
/* Call find_field_in_tables only to report the error */
638
find_field_in_tables(session, this,
639
context->first_name_resolution_table,
640
context->last_name_resolution_table,
641
reference, REPORT_ALL_ERRORS, true);
645
else if (ref != not_found_item)
650
/* Should have been checked in resolve_ref_in_select_and_group(). */
651
assert(*ref && (*ref)->fixed);
653
Here, a subset of actions performed by Item_ref::set_properties
654
is not enough. So we pass ptr to NULL into Item_[direct]_ref
655
constructor, so no initialization is performed, and call
659
*ref= NULL; // Don't call set_properties()
660
rf= (place == IN_HAVING ?
661
new Item_ref(context, ref, (char*) table_name,
662
(char*) field_name, alias_name_used) :
663
(!select->group_list.elements ?
664
new Item_direct_ref(context, ref, (char*) table_name,
665
(char*) field_name, alias_name_used) :
666
new Item_outer_ref(context, ref, (char*) table_name,
667
(char*) field_name, alias_name_used)));
672
if (place != IN_HAVING && select->group_list.elements)
674
outer_context->select_lex->inner_refs_list.push_back((Item_outer_ref*)rf);
675
((Item_outer_ref*)rf)->in_sum_func= session->lex->in_sum_func;
677
session->change_item_tree(reference, rf);
679
rf is Item_ref => never substitute other items (in this case)
680
during fix_fields() => we can use rf after fix_fields()
682
assert(!rf->fixed); // Assured by Item_ref()
683
if (rf->fix_fields(session, reference) || rf->check_cols(1))
686
mark_as_dependent(session, last_checked_context->select_lex,
687
context->select_lex, this,
693
mark_as_dependent(session, last_checked_context->select_lex,
695
this, (Item_ident*)*reference);
696
if (last_checked_context->select_lex->having_fix_field)
699
rf= new Item_ref(context,
700
(cached_table->db[0] ? cached_table->db : 0),
701
(char*) cached_table->alias, (char*) field_name);
704
session->change_item_tree(reference, rf);
706
rf is Item_ref => never substitute other items (in this case)
707
during fix_fields() => we can use rf after fix_fields()
709
assert(!rf->fixed); // Assured by Item_ref()
710
if (rf->fix_fields(session, reference) || rf->check_cols(1))
720
Resolve the name of a column reference.
722
The method resolves the column reference represented by 'this' as a column
723
present in one of: FROM clause, SELECT clause, GROUP BY clause of a query
724
Q, or in outer queries that contain Q.
726
The name resolution algorithm used is (where [T_j] is an optional table
727
name that qualifies the column name):
730
resolve_column_reference([T_j].col_ref_i)
732
search for a column or derived column named col_ref_i
733
[in table T_j] in the FROM clause of Q;
735
if such a column is NOT found AND // Lookup in outer queries.
736
there are outer queries
738
for each outer query Q_k beginning from the inner-most one
740
search for a column or derived column named col_ref_i
741
[in table T_j] in the FROM clause of Q_k;
743
if such a column is not found
744
Search for a column or derived column named col_ref_i
745
[in table T_j] in the SELECT and GROUP clauses of Q_k.
751
Notice that compared to Item_ref::fix_fields, here we first search the FROM
752
clause, and then we search the SELECT and GROUP BY clauses.
754
@param[in] session current thread
755
@param[in,out] reference view column if this item was resolved to a
764
bool Item_field::fix_fields(Session *session, Item **reference)
767
Field *from_field= (Field *)not_found_field;
768
bool outer_fixed= false;
770
if (!field) // If field is not checked
773
In case of view, find_field_in_tables() write pointer to view field
774
expression to 'reference', i.e. it substitute that expression instead
777
if ((from_field= find_field_in_tables(session, this,
778
context->first_name_resolution_table,
779
context->last_name_resolution_table,
781
session->lex->use_only_table_context ?
783
IGNORE_EXCEPT_NON_UNIQUE, true)) ==
787
/* Look up in current select's item_list to find aliased fields */
788
if (session->lex->current_select->is_item_list_lookup)
791
enum_resolution_type resolution;
792
Item** res= find_item_in_list(session,
793
this, session->lex->current_select->item_list,
794
&counter, REPORT_EXCEPT_NOT_FOUND,
798
if (resolution == RESOLVED_AGAINST_ALIAS)
799
alias_name_used= true;
800
if (res != (Item **)not_found_item)
802
if ((*res)->type() == Item::FIELD_ITEM)
805
It's an Item_field referencing another Item_field in the select
807
Use the field from the Item_field in the select list and leave
808
the Item_field instance in place.
811
Field *new_field= (*((Item_field**)res))->field;
813
if (new_field == NULL)
815
/* The column to which we link isn't valid. */
816
my_error(ER_BAD_FIELD_ERROR, MYF(0), (*res)->name,
817
current_session->where);
821
set_field(new_field);
827
It's not an Item_field in the select list so we must make a new
828
Item_ref to point to the Item in the select list and replace the
829
Item_field created by the parser with the new Item_ref.
831
Item_ref *rf= new Item_ref(context, db_name,table_name,field_name);
834
session->change_item_tree(reference, rf);
836
Because Item_ref never substitutes itself with other items
837
in Item_ref::fix_fields(), we can safely use the original
838
pointer to it even after fix_fields()
840
return rf->fix_fields(session, reference) || rf->check_cols(1);
844
if ((ret= fix_outer_field(session, &from_field, reference)) < 0)
848
goto mark_non_agg_field;
850
else if (!from_field)
853
if (!outer_fixed && cached_table && cached_table->select_lex &&
854
context->select_lex &&
855
cached_table->select_lex != context->select_lex)
858
if ((ret= fix_outer_field(session, &from_field, reference)) < 0)
862
goto mark_non_agg_field;
866
if it is not expression from merged VIEW we will set this field.
868
We can leave expression substituted from view for next PS/SP rexecution
869
(i.e. do not register this substitution for reverting on cleanup()
870
(register_item_tree_changing())), because this subtree will be
871
fix_field'ed during setup_tables()->setup_underlying() (i.e. before
872
all other expressions of query, and references on tables which do
873
not present in query will not make problems.
875
Also we suppose that view can't be changed during PS/SP life.
877
if (from_field == view_ref_found)
880
set_field(from_field);
881
if (session->lex->in_sum_func &&
882
session->lex->in_sum_func->nest_level ==
883
session->lex->current_select->nest_level)
885
set_if_bigger(session->lex->in_sum_func->max_arg_level,
886
session->lex->current_select->nest_level);
889
else if (session->mark_used_columns != MARK_COLUMNS_NONE)
891
Table *table= field->getTable();
892
MyBitmap *current_bitmap, *other_bitmap;
893
if (session->mark_used_columns == MARK_COLUMNS_READ)
895
current_bitmap= table->read_set;
896
other_bitmap= table->write_set;
900
current_bitmap= table->write_set;
901
other_bitmap= table->read_set;
903
if (! current_bitmap->testAndSet(field->field_index))
905
if (! other_bitmap->isBitSet(field->field_index))
907
/* First usage of column */
908
table->used_fields++; // Used to optimize loops
909
table->covering_keys&= field->part_of_key;
918
context->process_error(session);
922
Item *Item_field::safe_charset_converter(const CHARSET_INFO * const tocs)
925
return Item::safe_charset_converter(tocs);
929
void Item_field::cleanup()
931
Item_ident::cleanup();
933
Even if this object was created by direct link to field in setup_wild()
934
it will be linked correctly next time by name of field and table alias.
935
I.e. we can drop 'field'.
937
field= result_field= 0;
943
bool Item_field::result_as_int64_t()
945
return field->can_be_compared_as_int64_t();
950
Find a field among specified multiple equalities.
952
The function first searches the field among multiple equalities
953
of the current level (in the cond_equal->current_level list).
954
If it fails, it continues searching in upper levels accessed
955
through a pointer cond_equal->upper_levels.
956
The search terminates as soon as a multiple equality containing
959
@param cond_equal reference to list of multiple equalities where
960
the field (this object) is to be looked for
963
- First Item_equal containing the field, if success
967
Item_equal *Item_field::find_item_equal(COND_EQUAL *cond_equal)
972
List_iterator_fast<Item_equal> li(cond_equal->current_level);
975
if (item->contains(field))
979
The field is not found in any of the multiple equalities
980
of the current level. Look for it in upper levels
982
cond_equal= cond_equal->upper_levels;
989
Check whether a field can be substituted by an equal item.
991
The function checks whether a substitution of the field
992
occurrence for an equal item is valid.
994
@param arg *arg != NULL <-> the field is in the context where
995
substitution for an equal item is valid
998
The following statement is not always true:
1002
This means substitution of an item for an equal item not always
1003
yields an equavalent condition. Here's an example:
1006
(LENGTH('a')=1) != (LENGTH('a ')=2)
1008
Such a substitution is surely valid if either the substituted
1009
field is not of a STRING type or if it is an argument of
1010
a comparison predicate.
1013
true substitution is valid
1018
bool Item_field::subst_argument_checker(unsigned char **arg)
1020
return (result_type() != STRING_RESULT) || (*arg);
1025
Set a pointer to the multiple equality the field reference belongs to
1028
The function looks for a multiple equality containing the field item
1029
among those referenced by arg.
1030
In the case such equality exists the function does the following.
1031
If the found multiple equality contains a constant, then the field
1032
reference is substituted for this constant, otherwise it sets a pointer
1033
to the multiple equality in the field item.
1036
@param arg reference to list of multiple equalities where
1037
the field (this object) is to be looked for
1040
This function is supposed to be called as a callback parameter in calls
1041
of the compile method.
1044
- pointer to the replacing constant item, if the field item was substituted
1045
- pointer to the field item, otherwise.
1048
Item *Item_field::equal_fields_propagator(unsigned char *arg)
1052
item_equal= find_item_equal((COND_EQUAL *) arg);
1055
item= item_equal->get_const();
1057
Disable const propagation for items used in different comparison contexts.
1058
This must be done because, for example, Item_hex_string->val_int() is not
1059
the same as (Item_hex_string->val_str() in BINARY column)->val_int().
1060
We cannot simply disable the replacement in a particular context (
1061
e.g. <bin_col> = <int_col> AND <bin_col> = <hex_string>) since
1062
Items don't know the context they are in and there are functions like
1063
IF (<hex_string>, 'yes', 'no').
1064
The same problem occurs when comparing a DATE/TIME field with a
1065
DATE/TIME represented as an int and as a string.
1068
(cmp_context != (Item_result)-1 && item->cmp_context != cmp_context))
1076
Mark the item to not be part of substitution if it's not a binary item.
1078
See comments in Arg_comparator::set_compare_func() for details.
1081
bool Item_field::set_no_const_sub(unsigned char *)
1083
if (field->charset() != &my_charset_bin)
1090
Replace an Item_field for an equal Item_field that evaluated earlier
1093
The function returns a pointer to an item that is taken from
1094
the very beginning of the item_equal list which the Item_field
1095
object refers to (belongs to) unless item_equal contains a constant
1096
item. In this case the function returns this constant item,
1097
(if the substitution does not require conversion).
1098
If the Item_field object does not refer any Item_equal object
1099
'this' is returned .
1101
@param arg a dummy parameter, is not used here
1105
This function is supposed to be called as a callback parameter in calls
1106
of the thransformer method.
1109
- pointer to a replacement Item_field if there is a better equal item or
1110
a pointer to a constant equal item;
1114
Item *Item_field::replace_equal_field(unsigned char *)
1118
Item *const_item_ptr= item_equal->get_const();
1121
if (cmp_context != (Item_result)-1 &&
1122
const_item_ptr->cmp_context != cmp_context)
1124
return const_item_ptr;
1126
Item_field *subst= item_equal->get_first();
1127
if (subst && !field->eq(subst->field))
1134
uint32_t Item_field::max_disp_length()
1136
return field->max_display_length();
1141
void Item_field::make_field(SendField *tmp_field)
1143
field->make_field(tmp_field);
1144
assert(tmp_field->table_name != 0);
1146
tmp_field->col_name=name; // Use user supplied name
1148
tmp_field->table_name= table_name;
1150
tmp_field->db_name= db_name;
1155
Set a field's value from a item.
1158
void Item_field::save_org_in_field(Field *to)
1160
if (field->is_null())
1163
set_field_to_null_with_conversions(to, 1);
1168
field_conv(to,field);
1173
int Item_field::save_in_field(Field *to, bool no_conversions)
1176
if (result_field->is_null())
1179
res= set_field_to_null_with_conversions(to, no_conversions);
1184
res= field_conv(to,result_field);
1191
bool Item_field::send(plugin::Client *client, String *)
1193
return client->store(result_field);
1197
void Item_field::update_null_value()
1200
need to set no_errors to prevent warnings about type conversion
1203
Session *session= field->getTable()->in_use;
1206
no_errors= session->no_errors;
1207
session->no_errors= 1;
1208
Item::update_null_value();
1209
session->no_errors= no_errors;
1214
Add the field to the select list and substitute it for the reference to
1218
Item_field::update_value_transformer()
1219
select_arg current select
1222
If the field doesn't belong to the table being inserted into then it is
1223
added to the select list, pointer to it is stored in the ref_pointer_array
1224
of the select and the field itself is substituted for the Item_ref object.
1225
This is done in order to get correct values from update fields that
1226
belongs to the SELECT part in the INSERT .. SELECT .. ON DUPLICATE KEY
1231
ref if all conditions are met
1232
this field otherwise
1235
Item *Item_field::update_value_transformer(unsigned char *select_arg)
1237
Select_Lex *select= (Select_Lex*)select_arg;
1240
if (field->getTable() != select->context.table_list->table)
1242
List<Item> *all_fields= &select->join->all_fields;
1243
Item **ref_pointer_array= select->ref_pointer_array;
1244
int el= all_fields->elements;
1247
ref_pointer_array[el]= (Item*)this;
1248
all_fields->push_front((Item*)this);
1249
ref= new Item_ref(&select->context, ref_pointer_array + el,
1250
table_name, field_name);
1257
void Item_field::print(String *str, enum_query_type query_type)
1259
if (field && field->getTable()->const_table)
1261
char buff[MAX_FIELD_WIDTH];
1262
String tmp(buff,sizeof(buff),str->charset());
1263
field->val_str(&tmp);
1264
if (field->is_null()) {
1265
str->append("NULL");
1274
Item_ident::print(str, query_type);
1278
} /* namespace drizzled */