mirror of
				https://github.com/MariaDB/server.git
				synced 2025-10-31 02:46:29 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			201 lines
		
	
	
	
		
			4.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			201 lines
		
	
	
	
		
			4.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|    Copyright (c) 2002, 2011, Oracle and/or its affiliates.
 | |
|    Copyright (c) 2011, 2020, MariaDB Corporation.
 | |
| 
 | |
|    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; version 2 of the License.
 | |
| 
 | |
|    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.
 | |
| 
 | |
|    You should have received a copy of the GNU General Public License
 | |
|    along with this program; if not, write to the Free Software
 | |
|    Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1335  USA */
 | |
| 
 | |
| #include "mariadb.h"
 | |
| #include "sql_priv.h"
 | |
| /*
 | |
|   It is necessary to include set_var.h instead of item.h because there
 | |
|   are dependencies on include order for set_var.h and item.h. This
 | |
|   will be resolved later.
 | |
| */
 | |
| #include "sql_class.h"                          // THD, set_var.h: THD
 | |
| #include "set_var.h"
 | |
| 
 | |
| void Item_row::illegal_method_call(const char *method)
 | |
| {
 | |
|   DBUG_ENTER("Item_row::illegal_method_call");
 | |
|   DBUG_PRINT("error", ("!!! %s method was called for row item", method));
 | |
|   DBUG_ASSERT(0);
 | |
|   my_error(ER_OPERAND_COLUMNS, MYF(0), 1);
 | |
|   DBUG_VOID_RETURN;
 | |
| }
 | |
| 
 | |
| bool Item_row::fix_fields(THD *thd, Item **ref)
 | |
| {
 | |
|   DBUG_ASSERT(fixed() == 0);
 | |
|   null_value= 0;
 | |
|   base_flags&= ~item_base_t::MAYBE_NULL;
 | |
| 
 | |
|   Item **arg, **arg_end;
 | |
|   for (arg= args, arg_end= args + arg_count; arg != arg_end ; arg++)
 | |
|   {
 | |
|     if ((*arg)->fix_fields_if_needed(thd, arg))
 | |
|       return TRUE;
 | |
|     // we can't assign 'item' before, because fix_fields() can change arg
 | |
|     Item *item= *arg;
 | |
|     used_tables_cache |= item->used_tables();
 | |
|     const_item_cache&= item->const_item() && !with_null;
 | |
|     not_null_tables_cache|= item->not_null_tables();
 | |
| 
 | |
|     if (const_item_cache)
 | |
|     {
 | |
|       if (item->cols() > 1)
 | |
| 	with_null|= item->null_inside();
 | |
|       else
 | |
|       {
 | |
| 	if (item->is_null())
 | |
|           with_null|= 1;
 | |
|       }
 | |
|     }
 | |
|     base_flags|= (item->base_flags & item_base_t::MAYBE_NULL);
 | |
|     with_flags|= item->with_flags;
 | |
|   }
 | |
|   base_flags|= item_base_t::FIXED;
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| 
 | |
| bool
 | |
| Item_row::eval_not_null_tables(void *opt_arg)
 | |
| {
 | |
|   Item **arg,**arg_end;
 | |
|   not_null_tables_cache= 0;
 | |
|   if (arg_count)
 | |
|   {		
 | |
|     for (arg= args, arg_end= args + arg_count; arg != arg_end ; arg++)
 | |
|     {
 | |
|       not_null_tables_cache|= (*arg)->not_null_tables();
 | |
|     }
 | |
|   }
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| 
 | |
| bool
 | |
| Item_row::find_not_null_fields(table_map allowed)
 | |
| {
 | |
|   if (~allowed & used_tables())
 | |
|     return false;
 | |
| 
 | |
|   Item **arg,**arg_end;
 | |
|   if (arg_count)
 | |
|   {
 | |
|     for (arg= args, arg_end= args + arg_count; arg != arg_end ; arg++)
 | |
|     {
 | |
|       if (!(*arg)->find_not_null_fields(allowed))
 | |
|         continue;
 | |
|     }
 | |
|   }
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| 
 | |
| void Item_row::cleanup()
 | |
| {
 | |
|   DBUG_ENTER("Item_row::cleanup");
 | |
| 
 | |
|   Item_fixed_hybrid::cleanup();
 | |
|   /* Reset to the original values */
 | |
|   used_tables_and_const_cache_init();
 | |
|   with_null= 0;
 | |
| 
 | |
|   DBUG_VOID_RETURN;
 | |
| }
 | |
| 
 | |
| 
 | |
| void Item_row::split_sum_func(THD *thd, Ref_ptr_array ref_pointer_array,
 | |
|                               List<Item> &fields, uint flags)
 | |
| {
 | |
|   Item **arg, **arg_end;
 | |
|   for (arg= args, arg_end= args + arg_count; arg != arg_end ; arg++)
 | |
|     (*arg)->split_sum_func2(thd, ref_pointer_array, fields, arg,
 | |
|                             flags | SPLIT_SUM_SKIP_REGISTERED);
 | |
| }
 | |
| 
 | |
| 
 | |
| void Item_row::fix_after_pullout(st_select_lex *new_parent, Item **ref,
 | |
|                                  bool merge)
 | |
| {
 | |
|   used_tables_and_const_cache_init();
 | |
|   not_null_tables_cache= 0;
 | |
|   for (uint i= 0; i < arg_count; i++)
 | |
|   {
 | |
|     args[i]->fix_after_pullout(new_parent, &args[i], merge);
 | |
|     used_tables_and_const_cache_join(args[i]);
 | |
|     not_null_tables_cache|= args[i]->not_null_tables();
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| bool Item_row::check_cols(uint c)
 | |
| {
 | |
|   if (c != arg_count)
 | |
|   {
 | |
|     my_error(ER_OPERAND_COLUMNS, MYF(0), c);
 | |
|     return 1;
 | |
|   }
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| void Item_row::print(String *str, enum_query_type query_type)
 | |
| {
 | |
|   str->append('(');
 | |
|   for (uint i= 0; i < arg_count; i++)
 | |
|   {
 | |
|     if (i)
 | |
|       str->append(',');
 | |
|     args[i]->print(str, query_type);
 | |
|   }
 | |
|   str->append(')');
 | |
| }
 | |
| 
 | |
| 
 | |
| Item *Item_row::transform(THD *thd, Item_transformer transformer, uchar *arg)
 | |
| {
 | |
|   DBUG_ASSERT(!thd->stmt_arena->is_stmt_prepare());
 | |
| 
 | |
|   if (transform_args(thd, transformer, arg))
 | |
|     return 0;
 | |
|   return (this->*transformer)(thd, arg);
 | |
| }
 | |
| 
 | |
| void Item_row::bring_value()
 | |
| {
 | |
|   for (uint i= 0; i < arg_count; i++)
 | |
|     args[i]->bring_value();
 | |
| }
 | |
| 
 | |
| 
 | |
| Item* Item_row::do_build_clone(THD *thd) const
 | |
| {
 | |
|   Item **copy_args= static_cast<Item**>
 | |
|     (alloc_root(thd->mem_root, sizeof(Item*) * arg_count));
 | |
|   if (unlikely(!copy_args))
 | |
|     return 0;
 | |
|   for (uint i= 0; i < arg_count; i++)
 | |
|   {
 | |
|     Item *arg_clone= args[i]->build_clone(thd);
 | |
|     if (!arg_clone)
 | |
|       return 0;
 | |
|     copy_args[i]= arg_clone;
 | |
|   }
 | |
|   Item_row *copy= (Item_row *) get_copy(thd);
 | |
|   if (unlikely(!copy))
 | |
|     return 0;
 | |
|   copy->args= copy_args;
 | |
|   return copy;
 | |
| }
 | 
