Commit 74deaae9 authored by Tatiana A. Nurnberg's avatar Tatiana A. Nurnberg

Bug#43508: Renaming timestamp or date column triggers table copy

Altering a table to update a column with types DATE or TIMESTAMP
would incorrectly be seen as a significant change that necessitates
a slow copy+rename operation instead of a fast update.

There were two problems:

The character set is magically set for TIMESTAMP to be "binary",
but that was done too deep in field use code for ALTER TABLE to
know of it.  Now, put that in the constructor for Field_timestamp.

Also, when we set the character set for the new replacement/
comparison field, also raise the "binary" field flag that tells us
we should compare it exactly.  That is necessary to match the old
stored definition.

Next is the problem that the default length for TIMESTAMP and DATE
fields is different than the length read from the .frm .  The
compressed size is written to the file, but the human-readable,
part-delimited length is used as default length.  IIRC, for
timestamp it was 19!=14, and for date it was 8!=10.  Length
mismatch causes a table copy.

Also, clean up a place where a comparison function alters one of its
parameters and replace it with an assertion of the condition it
mutates.
parent e2ac8c07
/* Copyright 2000-2008 MySQL AB, 2008 Sun Microsystems, Inc. /* Copyright 2000-2008 MySQL AB, 2008-2009 Sun Microsystems, Inc.
This program is free software; you can redistribute it and/or modify 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 it under the terms of the GNU General Public License as published by
...@@ -4723,6 +4723,7 @@ Field_timestamp::Field_timestamp(uchar *ptr_arg, uint32 len_arg, ...@@ -4723,6 +4723,7 @@ Field_timestamp::Field_timestamp(uchar *ptr_arg, uint32 len_arg,
{ {
/* For 4.0 MYD and 4.0 InnoDB compatibility */ /* For 4.0 MYD and 4.0 InnoDB compatibility */
flags|= ZEROFILL_FLAG | UNSIGNED_FLAG; flags|= ZEROFILL_FLAG | UNSIGNED_FLAG;
field_charset= &my_charset_bin;
if (!share->timestamp_field && unireg_check != NONE) if (!share->timestamp_field && unireg_check != NONE)
{ {
/* This timestamp has auto-update */ /* This timestamp has auto-update */
...@@ -4743,6 +4744,7 @@ Field_timestamp::Field_timestamp(bool maybe_null_arg, ...@@ -4743,6 +4744,7 @@ Field_timestamp::Field_timestamp(bool maybe_null_arg,
{ {
/* For 4.0 MYD and 4.0 InnoDB compatibility */ /* For 4.0 MYD and 4.0 InnoDB compatibility */
flags|= ZEROFILL_FLAG | UNSIGNED_FLAG; flags|= ZEROFILL_FLAG | UNSIGNED_FLAG;
field_charset= &my_charset_bin;
if (unireg_check != TIMESTAMP_DN_FIELD) if (unireg_check != TIMESTAMP_DN_FIELD)
flags|= ON_UPDATE_NOW_FLAG; flags|= ON_UPDATE_NOW_FLAG;
} }
...@@ -6504,20 +6506,9 @@ uint Field::is_equal(Create_field *new_field) ...@@ -6504,20 +6506,9 @@ uint Field::is_equal(Create_field *new_field)
} }
/* If one of the fields is binary and the other one isn't return 1 else 0 */
bool Field_str::compare_str_field_flags(Create_field *new_field, uint32 flag_arg)
{
return (((new_field->flags & (BINCMP_FLAG | BINARY_FLAG)) &&
!(flag_arg & (BINCMP_FLAG | BINARY_FLAG))) ||
(!(new_field->flags & (BINCMP_FLAG | BINARY_FLAG)) &&
(flag_arg & (BINCMP_FLAG | BINARY_FLAG))));
}
uint Field_str::is_equal(Create_field *new_field) uint Field_str::is_equal(Create_field *new_field)
{ {
if (compare_str_field_flags(new_field, flags)) if (field_flags_are_binary() != new_field->field_flags_are_binary())
return 0; return 0;
return ((new_field->sql_type == real_type()) && return ((new_field->sql_type == real_type()) &&
...@@ -8283,7 +8274,7 @@ uint Field_blob::max_packed_col_length(uint max_length) ...@@ -8283,7 +8274,7 @@ uint Field_blob::max_packed_col_length(uint max_length)
uint Field_blob::is_equal(Create_field *new_field) uint Field_blob::is_equal(Create_field *new_field)
{ {
if (compare_str_field_flags(new_field, flags)) if (field_flags_are_binary() != new_field->field_flags_are_binary())
return 0; return 0;
return ((new_field->sql_type == get_blob_type_from_length(max_data_length())) return ((new_field->sql_type == get_blob_type_from_length(max_data_length()))
...@@ -9569,7 +9560,7 @@ bool Create_field::init(THD *thd, char *fld_name, enum_field_types fld_type, ...@@ -9569,7 +9560,7 @@ bool Create_field::init(THD *thd, char *fld_name, enum_field_types fld_type,
} }
if (length == 0) if (length == 0)
fld_length= 0; /* purecov: inspected */ fld_length= NULL; /* purecov: inspected */
} }
sign_len= fld_type_modifier & UNSIGNED_FLAG ? 0 : 1; sign_len= fld_type_modifier & UNSIGNED_FLAG ? 0 : 1;
...@@ -9721,8 +9712,7 @@ bool Create_field::init(THD *thd, char *fld_name, enum_field_types fld_type, ...@@ -9721,8 +9712,7 @@ bool Create_field::init(THD *thd, char *fld_name, enum_field_types fld_type,
case MYSQL_TYPE_TIMESTAMP: case MYSQL_TYPE_TIMESTAMP:
if (fld_length == NULL) if (fld_length == NULL)
{ {
/* Compressed date YYYYMMDDHHMMSS */ length= MAX_DATETIME_WIDTH;
length= MAX_DATETIME_COMPRESSED_WIDTH;
} }
else if (length != MAX_DATETIME_WIDTH) else if (length != MAX_DATETIME_WIDTH)
{ {
...@@ -9787,7 +9777,7 @@ bool Create_field::init(THD *thd, char *fld_name, enum_field_types fld_type, ...@@ -9787,7 +9777,7 @@ bool Create_field::init(THD *thd, char *fld_name, enum_field_types fld_type,
sql_type= MYSQL_TYPE_NEWDATE; sql_type= MYSQL_TYPE_NEWDATE;
/* fall trough */ /* fall trough */
case MYSQL_TYPE_NEWDATE: case MYSQL_TYPE_NEWDATE:
length= 10; length= MAX_DATE_WIDTH;
break; break;
case MYSQL_TYPE_TIME: case MYSQL_TYPE_TIME:
length= 10; length= 10;
...@@ -9868,6 +9858,17 @@ bool Create_field::init(THD *thd, char *fld_name, enum_field_types fld_type, ...@@ -9868,6 +9858,17 @@ bool Create_field::init(THD *thd, char *fld_name, enum_field_types fld_type,
DBUG_RETURN(TRUE); DBUG_RETURN(TRUE);
} }
switch (fld_type) {
case MYSQL_TYPE_DATE:
case MYSQL_TYPE_NEWDATE:
case MYSQL_TYPE_TIME:
case MYSQL_TYPE_DATETIME:
case MYSQL_TYPE_TIMESTAMP:
charset= &my_charset_bin;
flags|= BINCMP_FLAG;
default: break;
}
DBUG_RETURN(FALSE); /* success */ DBUG_RETURN(FALSE); /* success */
} }
...@@ -9976,16 +9977,6 @@ Field *make_field(TABLE_SHARE *share, uchar *ptr, uint32 field_length, ...@@ -9976,16 +9977,6 @@ Field *make_field(TABLE_SHARE *share, uchar *ptr, uint32 field_length,
null_bit= ((uchar) 1) << null_bit; null_bit= ((uchar) 1) << null_bit;
} }
switch (field_type) {
case MYSQL_TYPE_DATE:
case MYSQL_TYPE_NEWDATE:
case MYSQL_TYPE_TIME:
case MYSQL_TYPE_DATETIME:
case MYSQL_TYPE_TIMESTAMP:
field_charset= &my_charset_bin;
default: break;
}
if (f_is_alpha(pack_flag)) if (f_is_alpha(pack_flag))
{ {
if (!f_is_packed(pack_flag)) if (!f_is_packed(pack_flag))
......
/* Copyright 2000-2008 MySQL AB, 2008 Sun Microsystems, Inc. /* Copyright 2000-2008 MySQL AB, 2008, 2009 Sun Microsystems, Inc.
This program is free software; you can redistribute it and/or modify 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 it under the terms of the GNU General Public License as published by
...@@ -603,6 +603,12 @@ class Field ...@@ -603,6 +603,12 @@ class Field
handle_int64(to, from, low_byte_first_from, table->s->db_low_byte_first); handle_int64(to, from, low_byte_first_from, table->s->db_low_byte_first);
return from + sizeof(int64); return from + sizeof(int64);
} }
bool field_flags_are_binary()
{
return (flags & (BINCMP_FLAG | BINARY_FLAG)) != 0;
}
}; };
...@@ -658,7 +664,6 @@ class Field_str :public Field { ...@@ -658,7 +664,6 @@ class Field_str :public Field {
friend class Create_field; friend class Create_field;
my_decimal *val_decimal(my_decimal *); my_decimal *val_decimal(my_decimal *);
virtual bool str_needs_quotes() { return TRUE; } virtual bool str_needs_quotes() { return TRUE; }
bool compare_str_field_flags(Create_field *new_field, uint32 flags);
uint is_equal(Create_field *new_field); uint is_equal(Create_field *new_field);
}; };
...@@ -1268,12 +1273,12 @@ class Field_date :public Field_str { ...@@ -1268,12 +1273,12 @@ class Field_date :public Field_str {
Field_date(uchar *ptr_arg, uchar *null_ptr_arg, uchar null_bit_arg, Field_date(uchar *ptr_arg, uchar *null_ptr_arg, uchar null_bit_arg,
enum utype unireg_check_arg, const char *field_name_arg, enum utype unireg_check_arg, const char *field_name_arg,
CHARSET_INFO *cs) CHARSET_INFO *cs)
:Field_str(ptr_arg, 10, null_ptr_arg, null_bit_arg, :Field_str(ptr_arg, MAX_DATE_WIDTH, null_ptr_arg, null_bit_arg,
unireg_check_arg, field_name_arg, cs) unireg_check_arg, field_name_arg, cs)
{} {}
Field_date(bool maybe_null_arg, const char *field_name_arg, Field_date(bool maybe_null_arg, const char *field_name_arg,
CHARSET_INFO *cs) CHARSET_INFO *cs)
:Field_str((uchar*) 0,10, maybe_null_arg ? (uchar*) "": 0,0, :Field_str((uchar*) 0, MAX_DATE_WIDTH, maybe_null_arg ? (uchar*) "": 0,0,
NONE, field_name_arg, cs) {} NONE, field_name_arg, cs) {}
enum_field_types type() const { return MYSQL_TYPE_DATE;} enum_field_types type() const { return MYSQL_TYPE_DATE;}
enum ha_base_keytype key_type() const { return HA_KEYTYPE_ULONG_INT; } enum ha_base_keytype key_type() const { return HA_KEYTYPE_ULONG_INT; }
...@@ -1383,12 +1388,12 @@ class Field_datetime :public Field_str { ...@@ -1383,12 +1388,12 @@ class Field_datetime :public Field_str {
Field_datetime(uchar *ptr_arg, uchar *null_ptr_arg, uchar null_bit_arg, Field_datetime(uchar *ptr_arg, uchar *null_ptr_arg, uchar null_bit_arg,
enum utype unireg_check_arg, const char *field_name_arg, enum utype unireg_check_arg, const char *field_name_arg,
CHARSET_INFO *cs) CHARSET_INFO *cs)
:Field_str(ptr_arg, 19, null_ptr_arg, null_bit_arg, :Field_str(ptr_arg, MAX_DATETIME_WIDTH, null_ptr_arg, null_bit_arg,
unireg_check_arg, field_name_arg, cs) unireg_check_arg, field_name_arg, cs)
{} {}
Field_datetime(bool maybe_null_arg, const char *field_name_arg, Field_datetime(bool maybe_null_arg, const char *field_name_arg,
CHARSET_INFO *cs) CHARSET_INFO *cs)
:Field_str((uchar*) 0,19, maybe_null_arg ? (uchar*) "": 0,0, :Field_str((uchar*) 0, MAX_DATETIME_WIDTH, maybe_null_arg ? (uchar*) "": 0,0,
NONE, field_name_arg, cs) {} NONE, field_name_arg, cs) {}
enum_field_types type() const { return MYSQL_TYPE_DATETIME;} enum_field_types type() const { return MYSQL_TYPE_DATETIME;}
#ifdef HAVE_LONG_LONG #ifdef HAVE_LONG_LONG
...@@ -2061,6 +2066,11 @@ class Create_field :public Sql_alloc ...@@ -2061,6 +2066,11 @@ class Create_field :public Sql_alloc
Item *on_update_value, LEX_STRING *comment, char *change, Item *on_update_value, LEX_STRING *comment, char *change,
List<String> *interval_list, CHARSET_INFO *cs, List<String> *interval_list, CHARSET_INFO *cs,
uint uint_geom_type); uint uint_geom_type);
bool field_flags_are_binary()
{
return (flags & (BINCMP_FLAG | BINARY_FLAG)) != 0;
}
}; };
......
...@@ -2468,8 +2468,10 @@ mysql_prepare_create_table(THD *thd, HA_CREATE_INFO *create_info, ...@@ -2468,8 +2468,10 @@ mysql_prepare_create_table(THD *thd, HA_CREATE_INFO *create_info,
executing a prepared statement for the second time. executing a prepared statement for the second time.
*/ */
sql_field->length= sql_field->char_length; sql_field->length= sql_field->char_length;
if (!sql_field->charset)
if (sql_field->charset == NULL)
sql_field->charset= create_info->default_table_charset; sql_field->charset= create_info->default_table_charset;
/* /*
table_charset is set in ALTER TABLE if we want change character set table_charset is set in ALTER TABLE if we want change character set
for all varchar/char columns. for all varchar/char columns.
...@@ -5470,8 +5472,8 @@ compare_tables(TABLE *table, ...@@ -5470,8 +5472,8 @@ compare_tables(TABLE *table,
Field **f_ptr, *field; Field **f_ptr, *field;
uint changes= 0, tmp; uint changes= 0, tmp;
uint key_count; uint key_count;
List_iterator_fast<Create_field> new_field_it, tmp_new_field_it; List_iterator_fast<Create_field> tmp_new_field_it;
Create_field *new_field, *tmp_new_field; Create_field *tmp_new_field;
KEY_PART_INFO *key_part; KEY_PART_INFO *key_part;
KEY_PART_INFO *end; KEY_PART_INFO *end;
THD *thd= table->in_use; THD *thd= table->in_use;
...@@ -5497,6 +5499,9 @@ compare_tables(TABLE *table, ...@@ -5497,6 +5499,9 @@ compare_tables(TABLE *table,
pass it to mysql_prepare_create_table, then use the result pass it to mysql_prepare_create_table, then use the result
to evaluate possibility of fast ALTER TABLE, and then to evaluate possibility of fast ALTER TABLE, and then
destroy the copy. destroy the copy.
We shouldn't need to refer later in the function to alter_info
after this step.
*/ */
Alter_info tmp_alter_info(*alter_info, thd->mem_root); Alter_info tmp_alter_info(*alter_info, thd->mem_root);
uint db_options= 0; /* not used */ uint db_options= 0; /* not used */
...@@ -5508,6 +5513,7 @@ compare_tables(TABLE *table, ...@@ -5508,6 +5513,7 @@ compare_tables(TABLE *table,
table->file, key_info_buffer, table->file, key_info_buffer,
&key_count, 0)) &key_count, 0))
DBUG_RETURN(1); DBUG_RETURN(1);
/* Allocate result buffers. */ /* Allocate result buffers. */
if (! (*index_drop_buffer= if (! (*index_drop_buffer=
(uint*) thd->alloc(sizeof(uint) * table->s->keys)) || (uint*) thd->alloc(sizeof(uint) * table->s->keys)) ||
...@@ -5541,7 +5547,7 @@ compare_tables(TABLE *table, ...@@ -5541,7 +5547,7 @@ compare_tables(TABLE *table,
prior to 5.0 branch. prior to 5.0 branch.
See BUG#6236. See BUG#6236.
*/ */
if (table->s->fields != alter_info->create_list.elements || if (table->s->fields != tmp_alter_info.create_list.elements ||
table->s->db_type() != create_info->db_type || table->s->db_type() != create_info->db_type ||
table->s->tmp_table || table->s->tmp_table ||
create_info->used_fields & HA_CREATE_USED_ENGINE || create_info->used_fields & HA_CREATE_USED_ENGINE ||
...@@ -5550,7 +5556,7 @@ compare_tables(TABLE *table, ...@@ -5550,7 +5556,7 @@ compare_tables(TABLE *table,
(table->s->row_type != create_info->row_type) || (table->s->row_type != create_info->row_type) ||
create_info->used_fields & HA_CREATE_USED_PACK_KEYS || create_info->used_fields & HA_CREATE_USED_PACK_KEYS ||
create_info->used_fields & HA_CREATE_USED_MAX_ROWS || create_info->used_fields & HA_CREATE_USED_MAX_ROWS ||
(alter_info->flags & (ALTER_RECREATE | ALTER_FOREIGN_KEY)) || (tmp_alter_info.flags & (ALTER_RECREATE | ALTER_FOREIGN_KEY)) ||
order_num || order_num ||
!table->s->mysql_version || !table->s->mysql_version ||
(table->s->frm_version < FRM_VER_TRUE_VARCHAR && varchar)) (table->s->frm_version < FRM_VER_TRUE_VARCHAR && varchar))
...@@ -5563,22 +5569,19 @@ compare_tables(TABLE *table, ...@@ -5563,22 +5569,19 @@ compare_tables(TABLE *table,
Use transformed info to evaluate possibility of fast ALTER TABLE Use transformed info to evaluate possibility of fast ALTER TABLE
but use the preserved field to persist modifications. but use the preserved field to persist modifications.
*/ */
new_field_it.init(alter_info->create_list);
tmp_new_field_it.init(tmp_alter_info.create_list); tmp_new_field_it.init(tmp_alter_info.create_list);
/* /*
Go through fields and check if the original ones are compatible Go through fields and check if the original ones are compatible
with new table. with new table.
*/ */
for (f_ptr= table->field, new_field= new_field_it++,
tmp_new_field= tmp_new_field_it++;
for (f_ptr= table->field, tmp_new_field= tmp_new_field_it++;
(field= *f_ptr); (field= *f_ptr);
f_ptr++, new_field= new_field_it++, f_ptr++, tmp_new_field= tmp_new_field_it++)
tmp_new_field= tmp_new_field_it++)
{ {
/* Make sure we have at least the default charset in use. */ DBUG_ASSERT(tmp_new_field->charset != NULL);
if (!new_field->charset)
new_field->charset= create_info->default_table_charset;
/* Check that NULL behavior is same for old and new fields */ /* Check that NULL behavior is same for old and new fields */
if ((tmp_new_field->flags & NOT_NULL_FLAG) != if ((tmp_new_field->flags & NOT_NULL_FLAG) !=
......
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