mirror of
				https://github.com/MariaDB/server.git
				synced 2025-10-24 07:13:33 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			613 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			613 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|    Copyright (c) 2015  MariaDB Foundation.
 | |
| 
 | |
|  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 "sql_type.h"
 | |
| #include "sql_const.h"
 | |
| #include "sql_class.h"
 | |
| #include "item.h"
 | |
| #include "log.h"
 | |
| 
 | |
| static Type_handler_tiny        type_handler_tiny;
 | |
| static Type_handler_short       type_handler_short;
 | |
| static Type_handler_long        type_handler_long;
 | |
| static Type_handler_longlong    type_handler_longlong;
 | |
| static Type_handler_int24       type_handler_int24;
 | |
| static Type_handler_year        type_handler_year;
 | |
| static Type_handler_bit         type_handler_bit;
 | |
| static Type_handler_float       type_handler_float;
 | |
| static Type_handler_double      type_handler_double;
 | |
| static Type_handler_time        type_handler_time;
 | |
| static Type_handler_time2       type_handler_time2;
 | |
| static Type_handler_date        type_handler_date;
 | |
| static Type_handler_newdate     type_handler_newdate;
 | |
| static Type_handler_datetime    type_handler_datetime;
 | |
| static Type_handler_datetime2   type_handler_datetime2;
 | |
| static Type_handler_timestamp   type_handler_timestamp;
 | |
| static Type_handler_timestamp2  type_handler_timestamp2;
 | |
| static Type_handler_olddecimal  type_handler_olddecimal;
 | |
| static Type_handler_newdecimal  type_handler_newdecimal;
 | |
| static Type_handler_null        type_handler_null;
 | |
| static Type_handler_string      type_handler_string;
 | |
| static Type_handler_varchar     type_handler_varchar;
 | |
| static Type_handler_tiny_blob   type_handler_tiny_blob;
 | |
| static Type_handler_medium_blob type_handler_medium_blob;
 | |
| static Type_handler_long_blob   type_handler_long_blob;
 | |
| static Type_handler_blob        type_handler_blob;
 | |
| #ifdef HAVE_SPATIAL
 | |
| static Type_handler_geometry    type_handler_geometry;
 | |
| #endif
 | |
| static Type_handler_enum        type_handler_enum;
 | |
| static Type_handler_set         type_handler_set;
 | |
| 
 | |
| 
 | |
| /**
 | |
|   This method is used by:
 | |
|   - Item_user_var_as_out_param::field_type()
 | |
|   - Item_func_udf_str::field_type()
 | |
|   - Item_empty_string::make_field()
 | |
| 
 | |
|   TODO: type_handler_adjusted_to_max_octet_length() and string_type_handler()
 | |
|   provide very similar functionality, to properly choose between
 | |
|   VARCHAR/VARBINARY vs TEXT/BLOB variations taking into accoung maximum
 | |
|   possible octet length.
 | |
| 
 | |
|   We should probably get rid of either of them and use the same method
 | |
|   all around the code.
 | |
| */
 | |
| const Type_handler *
 | |
| Type_handler::string_type_handler(uint max_octet_length) const
 | |
| {
 | |
|   if (max_octet_length >= 16777216)
 | |
|     return &type_handler_long_blob;
 | |
|   else if (max_octet_length >= 65536)
 | |
|     return &type_handler_medium_blob;
 | |
|   return &type_handler_varchar;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   This method is used by:
 | |
|   - Item_sum_hybrid, e.g. MAX(item), MIN(item).
 | |
|   - Item_func_set_user_var
 | |
| */
 | |
| const Type_handler *
 | |
| Type_handler_string_result::type_handler_adjusted_to_max_octet_length(
 | |
|                                                         uint max_octet_length,
 | |
|                                                         CHARSET_INFO *cs) const
 | |
| {
 | |
|   if (max_octet_length / cs->mbmaxlen <= CONVERT_IF_BIGGER_TO_BLOB)
 | |
|     return &type_handler_varchar; // See also Item::too_big_for_varchar()
 | |
|   if (max_octet_length >= 16777216)
 | |
|     return &type_handler_long_blob;
 | |
|   else if (max_octet_length >= 65536)
 | |
|     return &type_handler_medium_blob;
 | |
|   return &type_handler_blob;
 | |
| }
 | |
| 
 | |
| 
 | |
| const Type_handler *
 | |
| Type_handler_hybrid_field_type::get_handler_by_result_type(Item_result type)
 | |
|                                                            const
 | |
| {
 | |
|   switch (type) {
 | |
|   case REAL_RESULT:       return &type_handler_double;
 | |
|   case INT_RESULT:        return &type_handler_longlong;
 | |
|   case DECIMAL_RESULT:    return &type_handler_newdecimal;
 | |
|   case STRING_RESULT:     return &type_handler_long_blob;
 | |
|   case TIME_RESULT:
 | |
|   case ROW_RESULT:
 | |
|     DBUG_ASSERT(0);
 | |
|   }
 | |
|   return &type_handler_string;
 | |
| }
 | |
| 
 | |
| 
 | |
| Type_handler_hybrid_field_type::Type_handler_hybrid_field_type()
 | |
|   :m_type_handler(&type_handler_double)
 | |
| {
 | |
| }
 | |
| 
 | |
| 
 | |
| const Type_handler *
 | |
| Type_handler::get_handler_by_field_type(enum_field_types type)
 | |
| {
 | |
|   switch (type) {
 | |
|   case MYSQL_TYPE_DECIMAL:     return &type_handler_olddecimal;
 | |
|   case MYSQL_TYPE_NEWDECIMAL:  return &type_handler_newdecimal;
 | |
|   case MYSQL_TYPE_TINY:        return &type_handler_tiny;
 | |
|   case MYSQL_TYPE_SHORT:       return &type_handler_short;
 | |
|   case MYSQL_TYPE_LONG:        return &type_handler_long;
 | |
|   case MYSQL_TYPE_LONGLONG:    return &type_handler_longlong;
 | |
|   case MYSQL_TYPE_INT24:       return &type_handler_int24;
 | |
|   case MYSQL_TYPE_YEAR:        return &type_handler_year;
 | |
|   case MYSQL_TYPE_BIT:         return &type_handler_bit;
 | |
|   case MYSQL_TYPE_FLOAT:       return &type_handler_float;
 | |
|   case MYSQL_TYPE_DOUBLE:      return &type_handler_double;
 | |
|   case MYSQL_TYPE_NULL:        return &type_handler_null;
 | |
|   case MYSQL_TYPE_VARCHAR:     return &type_handler_varchar;
 | |
|   case MYSQL_TYPE_TINY_BLOB:   return &type_handler_tiny_blob;
 | |
|   case MYSQL_TYPE_MEDIUM_BLOB: return &type_handler_medium_blob;
 | |
|   case MYSQL_TYPE_LONG_BLOB:   return &type_handler_long_blob;
 | |
|   case MYSQL_TYPE_BLOB:        return &type_handler_blob;
 | |
|   case MYSQL_TYPE_VAR_STRING:  return &type_handler_varchar; // Map to VARCHAR 
 | |
|   case MYSQL_TYPE_STRING:      return &type_handler_string;
 | |
|   case MYSQL_TYPE_ENUM:        return &type_handler_varchar; // Map to VARCHAR
 | |
|   case MYSQL_TYPE_SET:         return &type_handler_varchar; // Map to VARCHAR
 | |
|   case MYSQL_TYPE_GEOMETRY:
 | |
| #ifdef HAVE_SPATIAL
 | |
|     return &type_handler_geometry;
 | |
| #else
 | |
|     return NULL;
 | |
| #endif
 | |
|   case MYSQL_TYPE_TIMESTAMP:   return &type_handler_timestamp2;// Map to timestamp2
 | |
|   case MYSQL_TYPE_TIMESTAMP2:  return &type_handler_timestamp2;
 | |
|   case MYSQL_TYPE_DATE:        return &type_handler_newdate;   // Map to newdate
 | |
|   case MYSQL_TYPE_TIME:        return &type_handler_time2;     // Map to time2
 | |
|   case MYSQL_TYPE_TIME2:       return &type_handler_time2;
 | |
|   case MYSQL_TYPE_DATETIME:    return &type_handler_datetime2; // Map to datetime2
 | |
|   case MYSQL_TYPE_DATETIME2:   return &type_handler_datetime2;
 | |
|   case MYSQL_TYPE_NEWDATE:
 | |
|     /*
 | |
|       NEWDATE is actually a real_type(), not a field_type(),
 | |
|       but it's used around the code in field_type() context.
 | |
|       We should probably clean up the code not to use MYSQL_TYPE_NEWDATE
 | |
|       in field_type() context and add DBUG_ASSERT(0) here.
 | |
|     */
 | |
|     return &type_handler_newdate;
 | |
|   };
 | |
|   DBUG_ASSERT(0);
 | |
|   return &type_handler_string;
 | |
| }
 | |
| 
 | |
| 
 | |
| const Type_handler *
 | |
| Type_handler::get_handler_by_real_type(enum_field_types type)
 | |
| {
 | |
|   switch (type) {
 | |
|   case MYSQL_TYPE_DECIMAL:     return &type_handler_olddecimal;
 | |
|   case MYSQL_TYPE_NEWDECIMAL:  return &type_handler_newdecimal;
 | |
|   case MYSQL_TYPE_TINY:        return &type_handler_tiny;
 | |
|   case MYSQL_TYPE_SHORT:       return &type_handler_short;
 | |
|   case MYSQL_TYPE_LONG:        return &type_handler_long;
 | |
|   case MYSQL_TYPE_LONGLONG:    return &type_handler_longlong;
 | |
|   case MYSQL_TYPE_INT24:       return &type_handler_int24;
 | |
|   case MYSQL_TYPE_YEAR:        return &type_handler_year;
 | |
|   case MYSQL_TYPE_BIT:         return &type_handler_bit;
 | |
|   case MYSQL_TYPE_FLOAT:       return &type_handler_float;
 | |
|   case MYSQL_TYPE_DOUBLE:      return &type_handler_double;
 | |
|   case MYSQL_TYPE_NULL:        return &type_handler_null;
 | |
|   case MYSQL_TYPE_VARCHAR:     return &type_handler_varchar;
 | |
|   case MYSQL_TYPE_TINY_BLOB:   return &type_handler_tiny_blob;
 | |
|   case MYSQL_TYPE_MEDIUM_BLOB: return &type_handler_medium_blob;
 | |
|   case MYSQL_TYPE_LONG_BLOB:   return &type_handler_long_blob;
 | |
|   case MYSQL_TYPE_BLOB:        return &type_handler_blob;
 | |
|   case MYSQL_TYPE_VAR_STRING:
 | |
|     /*
 | |
|       VAR_STRING is actually a field_type(), not a real_type(),
 | |
|       but it's used around the code in real_type() context.
 | |
|       We should clean up the code and add DBUG_ASSERT(0) here.
 | |
|     */
 | |
|     return &type_handler_string;
 | |
|   case MYSQL_TYPE_STRING:      return &type_handler_string;
 | |
|   case MYSQL_TYPE_ENUM:        return &type_handler_enum;
 | |
|   case MYSQL_TYPE_SET:         return &type_handler_set;
 | |
|   case MYSQL_TYPE_GEOMETRY:
 | |
| #ifdef HAVE_SPATIAL
 | |
|     return &type_handler_geometry;
 | |
| #else
 | |
|     return NULL;
 | |
| #endif
 | |
|   case MYSQL_TYPE_TIMESTAMP:   return &type_handler_timestamp;
 | |
|   case MYSQL_TYPE_TIMESTAMP2:  return &type_handler_timestamp2;
 | |
|   case MYSQL_TYPE_DATE:        return &type_handler_date;
 | |
|   case MYSQL_TYPE_TIME:        return &type_handler_time;
 | |
|   case MYSQL_TYPE_TIME2:       return &type_handler_time2;
 | |
|   case MYSQL_TYPE_DATETIME:    return &type_handler_datetime;
 | |
|   case MYSQL_TYPE_DATETIME2:   return &type_handler_datetime2;
 | |
|   case MYSQL_TYPE_NEWDATE:     return &type_handler_newdate;
 | |
|   };
 | |
|   DBUG_ASSERT(0);
 | |
|   return &type_handler_string;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|   Create a DOUBLE field by default.
 | |
| */
 | |
| Field *
 | |
| Type_handler::make_num_distinct_aggregator_field(MEM_ROOT *mem_root,
 | |
|                                                  const Item *item) const
 | |
| {
 | |
|   return new(mem_root)
 | |
|          Field_double(NULL, item->max_length,
 | |
|                       (uchar *) (item->maybe_null ? "" : 0),
 | |
|                       item->maybe_null ? 1 : 0, Field::NONE,
 | |
|                       item->name, item->decimals, 0, item->unsigned_flag);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *
 | |
| Type_handler_float::make_num_distinct_aggregator_field(MEM_ROOT *mem_root,
 | |
|                                                        const Item *item)
 | |
|                                                        const
 | |
| {
 | |
|   return new(mem_root)
 | |
|          Field_float(NULL, item->max_length,
 | |
|                      (uchar *) (item->maybe_null ? "" : 0),
 | |
|                      item->maybe_null ? 1 : 0, Field::NONE,
 | |
|                      item->name, item->decimals, 0, item->unsigned_flag);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *
 | |
| Type_handler_decimal_result::make_num_distinct_aggregator_field(
 | |
|                                                             MEM_ROOT *mem_root,
 | |
|                                                             const Item *item)
 | |
|                                                             const
 | |
| {
 | |
|   DBUG_ASSERT(item->decimals <= DECIMAL_MAX_SCALE);
 | |
|   return new (mem_root)
 | |
|          Field_new_decimal(NULL, item->max_length,
 | |
|                            (uchar *) (item->maybe_null ? "" : 0),
 | |
|                            item->maybe_null ? 1 : 0, Field::NONE,
 | |
|                            item->name, item->decimals, 0, item->unsigned_flag);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *
 | |
| Type_handler_int_result::make_num_distinct_aggregator_field(MEM_ROOT *mem_root,
 | |
|                                                             const Item *item)
 | |
|                                                             const
 | |
| {
 | |
|   /**
 | |
|     Make a longlong field for all INT-alike types. It could create
 | |
|     smaller fields for TINYINT, SMALLINT, MEDIUMINT, INT though.
 | |
|   */
 | |
|   return new(mem_root)
 | |
|          Field_longlong(NULL, item->max_length,
 | |
|                         (uchar *) (item->maybe_null ? "" : 0),
 | |
|                         item->maybe_null ? 1 : 0, Field::NONE,
 | |
|                         item->name, 0, item->unsigned_flag);
 | |
| }
 | |
| 
 | |
| 
 | |
| /***********************************************************************/
 | |
| 
 | |
| #define TMPNAME ""
 | |
| 
 | |
| Field *Type_handler_tiny::make_conversion_table_field(TABLE *table,
 | |
|                                                       uint metadata,
 | |
|                                                       const Field *target)
 | |
|                                                       const
 | |
| {
 | |
|   /*
 | |
|     As we don't know if the integer was signed or not on the master,
 | |
|     assume we have same sign on master and slave.  This is true when not
 | |
|     using conversions so it should be true also when using conversions.
 | |
|   */
 | |
|   bool unsigned_flag= ((Field_num*) target)->unsigned_flag;
 | |
|   return new (table->in_use->mem_root)
 | |
|          Field_tiny(NULL, 4 /*max_length*/, (uchar *) "", 1, Field::NONE,
 | |
|                     TMPNAME, 0/*zerofill*/, unsigned_flag);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_short::make_conversion_table_field(TABLE *table,
 | |
|                                                        uint metadata,
 | |
|                                                        const Field *target)
 | |
|                                                        const
 | |
| {
 | |
|   bool unsigned_flag= ((Field_num*) target)->unsigned_flag;
 | |
|   return new (table->in_use->mem_root)
 | |
|          Field_short(NULL, 6 /*max_length*/, (uchar *) "", 1, Field::NONE,
 | |
|                      TMPNAME, 0/*zerofill*/, unsigned_flag);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_int24::make_conversion_table_field(TABLE *table,
 | |
|                                                        uint metadata,
 | |
|                                                        const Field *target)
 | |
|                                                        const
 | |
| {
 | |
|   bool unsigned_flag= ((Field_num*) target)->unsigned_flag;
 | |
|   return new (table->in_use->mem_root)
 | |
|          Field_medium(NULL, 9 /*max_length*/, (uchar *) "", 1, Field::NONE,
 | |
|                       TMPNAME, 0/*zerofill*/, unsigned_flag);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_long::make_conversion_table_field(TABLE *table,
 | |
|                                                       uint metadata,
 | |
|                                                       const Field *target)
 | |
|                                                       const
 | |
| {
 | |
|   bool unsigned_flag= ((Field_num*) target)->unsigned_flag;
 | |
|   return new (table->in_use->mem_root)
 | |
|          Field_long(NULL, 11 /*max_length*/, (uchar *) "", 1, Field::NONE,
 | |
|          TMPNAME, 0/*zerofill*/, unsigned_flag);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_longlong::make_conversion_table_field(TABLE *table,
 | |
|                                                           uint metadata,
 | |
|                                                           const Field *target)
 | |
|                                                           const
 | |
| {
 | |
|   bool unsigned_flag= ((Field_num*) target)->unsigned_flag;
 | |
|   return new (table->in_use->mem_root)
 | |
|          Field_longlong(NULL, 20 /*max_length*/,(uchar *) "", 1, Field::NONE,
 | |
|                         TMPNAME, 0/*zerofill*/, unsigned_flag);
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| Field *Type_handler_float::make_conversion_table_field(TABLE *table,
 | |
|                                                        uint metadata,
 | |
|                                                        const Field *target)
 | |
|                                                        const
 | |
| {
 | |
|   return new (table->in_use->mem_root)
 | |
|          Field_float(NULL, 12 /*max_length*/, (uchar *) "", 1, Field::NONE,
 | |
|                      TMPNAME, 0/*dec*/, 0/*zerofill*/, 0/*unsigned_flag*/);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_double::make_conversion_table_field(TABLE *table,
 | |
|                                                         uint metadata,
 | |
|                                                         const Field *target)
 | |
|                                                         const
 | |
| {
 | |
|   return new (table->in_use->mem_root)
 | |
|          Field_double(NULL, 22 /*max_length*/, (uchar *) "", 1, Field::NONE,
 | |
|                       TMPNAME, 0/*dec*/, 0/*zerofill*/, 0/*unsigned_flag*/);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_newdecimal::make_conversion_table_field(TABLE *table,
 | |
|                                                             uint metadata,
 | |
|                                                             const Field *target)
 | |
|                                                             const
 | |
| {
 | |
|   int  precision= metadata >> 8;
 | |
|   uint decimals= metadata & 0x00ff;
 | |
|   uint32 max_length= my_decimal_precision_to_length(precision, decimals, false);
 | |
|   DBUG_ASSERT(decimals <= DECIMAL_MAX_SCALE);
 | |
|   return new (table->in_use->mem_root)
 | |
|          Field_new_decimal(NULL, max_length, (uchar *) "", 1, Field::NONE,
 | |
|                            TMPNAME, decimals, 0/*zerofill*/, 0/*unsigned*/);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_olddecimal::make_conversion_table_field(TABLE *table,
 | |
|                                                             uint metadata,
 | |
|                                                             const Field *target)
 | |
|                                                             const
 | |
| {
 | |
|   sql_print_error("In RBR mode, Slave received incompatible DECIMAL field "
 | |
|                   "(old-style decimal field) from Master while creating "
 | |
|                   "conversion table. Please consider changing datatype on "
 | |
|                   "Master to new style decimal by executing ALTER command for"
 | |
|                   " column Name: %s.%s.%s.",
 | |
|                   target->table->s->db.str,
 | |
|                   target->table->s->table_name.str,
 | |
|                   target->field_name);
 | |
|   return NULL;
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_year::make_conversion_table_field(TABLE *table,
 | |
|                                                       uint metadata,
 | |
|                                                       const Field *target)
 | |
|                                                       const
 | |
| {
 | |
|   return new(table->in_use->mem_root)
 | |
|          Field_year(NULL, 4, (uchar *) "", 1, Field::NONE, TMPNAME);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_null::make_conversion_table_field(TABLE *table,
 | |
|                                                       uint metadata,
 | |
|                                                       const Field *target)
 | |
|                                                       const
 | |
| {
 | |
|   return new(table->in_use->mem_root)
 | |
|          Field_null(NULL, 0, Field::NONE, TMPNAME, target->charset());
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_timestamp::make_conversion_table_field(TABLE *table,
 | |
|                                                            uint metadata,
 | |
|                                                            const Field *target)
 | |
|                                                            const
 | |
| {
 | |
|   return new_Field_timestamp(table->in_use->mem_root, NULL, (uchar *) "", 1,
 | |
|                            Field::NONE, TMPNAME, table->s, target->decimals());
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_timestamp2::make_conversion_table_field(TABLE *table,
 | |
|                                                             uint metadata,
 | |
|                                                             const Field *target)
 | |
|                                                             const
 | |
| {
 | |
|   return new(table->in_use->mem_root)
 | |
|          Field_timestampf(NULL, (uchar *) "", 1, Field::NONE,
 | |
|                           TMPNAME, table->s, metadata);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_newdate::make_conversion_table_field(TABLE *table,
 | |
|                                                          uint metadata,
 | |
|                                                          const Field *target)
 | |
|                                                          const
 | |
| {
 | |
|   return new(table->in_use->mem_root)
 | |
|          Field_newdate(NULL, (uchar *) "", 1, Field::NONE, TMPNAME);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_date::make_conversion_table_field(TABLE *table,
 | |
|                                                       uint metadata,
 | |
|                                                       const Field *target)
 | |
|                                                       const
 | |
| {
 | |
|   return new(table->in_use->mem_root)
 | |
|          Field_date(NULL, (uchar *) "", 1, Field::NONE, TMPNAME);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_time::make_conversion_table_field(TABLE *table,
 | |
|                                                       uint metadata,
 | |
|                                                       const Field *target)
 | |
|                                                       const
 | |
| {
 | |
|   return new_Field_time(table->in_use->mem_root, NULL, (uchar *) "", 1,
 | |
|                         Field::NONE, TMPNAME, target->decimals());
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_time2::make_conversion_table_field(TABLE *table,
 | |
|                                                        uint metadata,
 | |
|                                                        const Field *target)
 | |
|                                                        const
 | |
| {
 | |
|   return new(table->in_use->mem_root)
 | |
|          Field_timef(NULL, (uchar *) "", 1, Field::NONE, TMPNAME, metadata);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_datetime::make_conversion_table_field(TABLE *table,
 | |
|                                                           uint metadata,
 | |
|                                                           const Field *target)
 | |
|                                                           const
 | |
| {
 | |
|   return new_Field_datetime(table->in_use->mem_root, NULL, (uchar *) "", 1,
 | |
|                             Field::NONE, TMPNAME, target->decimals());
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_datetime2::make_conversion_table_field(TABLE *table,
 | |
|                                                            uint metadata,
 | |
|                                                            const Field *target)
 | |
|                                                            const
 | |
| {
 | |
|   return new(table->in_use->mem_root)
 | |
|          Field_datetimef(NULL, (uchar *) "", 1,
 | |
|                          Field::NONE, TMPNAME, metadata);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_bit::make_conversion_table_field(TABLE *table,
 | |
|                                                      uint metadata,
 | |
|                                                      const Field *target)
 | |
|                                                      const
 | |
| {
 | |
|   DBUG_ASSERT((metadata & 0xff) <= 7);
 | |
|   uint32 max_length= 8 * (metadata >> 8U) + (metadata & 0x00ff);
 | |
|   return new(table->in_use->mem_root)
 | |
|          Field_bit_as_char(NULL, max_length, (uchar *) "", 1,
 | |
|                            Field::NONE, TMPNAME);
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_string::make_conversion_table_field(TABLE *table,
 | |
|                                                         uint metadata,
 | |
|                                                         const Field *target)
 | |
|                                                         const
 | |
| {
 | |
|   /* This is taken from Field_string::unpack. */
 | |
|   uint32 max_length= (((metadata >> 4) & 0x300) ^ 0x300) + (metadata & 0x00ff);
 | |
|   return new(table->in_use->mem_root)
 | |
|          Field_string(NULL, max_length, (uchar *) "", 1,
 | |
|                       Field::NONE, TMPNAME, target->charset());
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_varchar::make_conversion_table_field(TABLE *table,
 | |
|                                                          uint metadata,
 | |
|                                                          const Field *target)
 | |
|                                                          const
 | |
| {
 | |
|   return new(table->in_use->mem_root)
 | |
|          Field_varstring(NULL, metadata, HA_VARCHAR_PACKLENGTH(metadata),
 | |
|                          (uchar *) "", 1, Field::NONE, TMPNAME,
 | |
|                          table->s, target->charset());
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_blob_common::make_conversion_table_field(TABLE *table,
 | |
|                                                             uint metadata,
 | |
|                                                             const Field *target)
 | |
|                                                             const
 | |
| {
 | |
|   uint pack_length= metadata & 0x00ff;
 | |
|   if (pack_length < 1 || pack_length > 4)
 | |
|     return NULL; // Broken binary log?
 | |
|   return new(table->in_use->mem_root)
 | |
|          Field_blob(NULL, (uchar *) "", 1, Field::NONE, TMPNAME,
 | |
|                     table->s, pack_length, target->charset());
 | |
| }
 | |
| 
 | |
| 
 | |
| #ifdef HAVE_SPATIAL
 | |
| Field *Type_handler_geometry::make_conversion_table_field(TABLE *table,
 | |
|                                                           uint metadata,
 | |
|                                                           const Field *target)
 | |
|                                                           const
 | |
| {
 | |
|   DBUG_ASSERT(target->type() == MYSQL_TYPE_GEOMETRY);
 | |
|   /*
 | |
|     We do not do not update feature_gis statistics here:
 | |
|     status_var_increment(target->table->in_use->status_var.feature_gis);
 | |
|     as this is only a temporary field.
 | |
|     The statistics was already incremented when "target" was created.
 | |
|   */
 | |
|   return new(table->in_use->mem_root)
 | |
|          Field_geom(NULL, (uchar *) "", 1, Field::NONE, TMPNAME, table->s, 4,
 | |
|                     ((const Field_geom*) target)->geom_type,
 | |
|                     ((const Field_geom*) target)->srid);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| Field *Type_handler_enum::make_conversion_table_field(TABLE *table,
 | |
|                                                       uint metadata,
 | |
|                                                       const Field *target)
 | |
|                                                       const
 | |
| {
 | |
|   DBUG_ASSERT(target->type() == MYSQL_TYPE_STRING);
 | |
|   DBUG_ASSERT(target->real_type() == MYSQL_TYPE_ENUM);
 | |
|   return new(table->in_use->mem_root)
 | |
|          Field_enum(NULL, target->field_length,
 | |
|                     (uchar *) "", 1, Field::NONE, TMPNAME,
 | |
|                     metadata & 0x00ff/*pack_length()*/,
 | |
|                     ((const Field_enum*) target)->typelib, target->charset());
 | |
| }
 | |
| 
 | |
| 
 | |
| Field *Type_handler_set::make_conversion_table_field(TABLE *table,
 | |
|                                                      uint metadata,
 | |
|                                                      const Field *target)
 | |
|                                                      const
 | |
| {
 | |
|   DBUG_ASSERT(target->type() == MYSQL_TYPE_STRING);
 | |
|   DBUG_ASSERT(target->real_type() == MYSQL_TYPE_SET);
 | |
|   return new(table->in_use->mem_root)
 | |
|          Field_set(NULL, target->field_length,
 | |
|                    (uchar *) "", 1, Field::NONE, TMPNAME,
 | |
|                    metadata & 0x00ff/*pack_length()*/,
 | |
|                    ((const Field_enum*) target)->typelib, target->charset());
 | |
| }
 |