Commit d1a666d5 authored by monty@mysql.com's avatar monty@mysql.com

Cleanups during review stage

Added auto-correct of field length for enum/set tables for ALTER TABLE
This is becasue of a bug in previous MySQL 4.1 versions where the length for enum/set was set incorrectly after ALTER TABLE
parent facaf4f7
slave stop; stop slave;
drop table if exists t1,t2,t3,t4,t5,t6,t7,t8,t9; drop table if exists t1,t2,t3,t4,t5,t6,t7,t8,t9;
reset master; reset master;
reset slave; reset slave;
drop table if exists t1,t2,t3,t4,t5,t6,t7,t8,t9; drop table if exists t1,t2,t3,t4,t5,t6,t7,t8,t9;
slave start; start slave;
stop slave; stop slave;
create table t1(n int); create table t1(n int);
start slave; start slave;
......
...@@ -1693,3 +1693,15 @@ oe ...@@ -1693,3 +1693,15 @@ oe
ue ue
ss ss
DROP TABLE t1; DROP TABLE t1;
create table t1 (a enum ('Y','N') CHARACTER SET utf8 COLLATE utf8_bin);
insert into t1 values ('Y');
alter table t1 add b set ('Y','N') CHARACTER SET utf8 COLLATE utf8_bin;
alter table t1 add c enum ('Y','N') CHARACTER SET utf8 COLLATE utf8_bin;
select * from t1;
Catalog Database Table Table_alias Column Column_alias Name Type Length Max length Is_null Flags Decimals Charsetnr
def test t1 t1 a a 254 3 1 Y 384 0 8
def test t1 t1 b b 254 9 0 Y 2176 0 8
def test t1 t1 c c 254 3 0 Y 384 0 8
a b c
Y NULL NULL
drop table t1;
...@@ -471,4 +471,3 @@ execute stmt using @var, @var, @var; ...@@ -471,4 +471,3 @@ execute stmt using @var, @var, @var;
set @var=null; set @var=null;
select @var is null, @var is not null, @var; select @var is null, @var is not null, @var;
execute stmt using @var, @var, @var; execute stmt using @var, @var, @var;
...@@ -72,3 +72,16 @@ CREATE TABLE t1 (c enum('ae','oe','ue','ss') collate latin1_german2_ci); ...@@ -72,3 +72,16 @@ CREATE TABLE t1 (c enum('ae','oe','ue','ss') collate latin1_german2_ci);
INSERT INTO t1 VALUES (''),(''),(''),(''); INSERT INTO t1 VALUES (''),(''),(''),('');
SELECT * FROM t1; SELECT * FROM t1;
DROP TABLE t1; DROP TABLE t1;
#
# Test bug where enum fields where extended for each ALTER TABLE
#
create table t1 (a enum ('Y','N') CHARACTER SET utf8 COLLATE utf8_bin);
insert into t1 values ('Y');
alter table t1 add b set ('Y','N') CHARACTER SET utf8 COLLATE utf8_bin;
alter table t1 add c enum ('Y','N') CHARACTER SET utf8 COLLATE utf8_bin;
--enable_metadata
select * from t1;
--disable metadata
drop table t1;
...@@ -5842,25 +5842,47 @@ bool Field_num::eq_def(Field *field) ...@@ -5842,25 +5842,47 @@ bool Field_num::eq_def(Field *field)
void create_field::create_length_to_internal_length(void) void create_field::create_length_to_internal_length(void)
{ {
switch (sql_type) switch (sql_type) {
case MYSQL_TYPE_TINY_BLOB:
case MYSQL_TYPE_MEDIUM_BLOB:
case MYSQL_TYPE_LONG_BLOB:
case MYSQL_TYPE_BLOB:
case MYSQL_TYPE_VAR_STRING:
case MYSQL_TYPE_STRING:
length*= charset->mbmaxlen;
pack_length= calc_pack_length(sql_type == FIELD_TYPE_VAR_STRING ?
FIELD_TYPE_STRING : sql_type, length);
break;
#ifdef CORRECT_CODE_BUT_CANT_YET_BE_USED
case MYSQL_TYPE_ENUM:
case MYSQL_TYPE_SET:
length*= charset->mbmaxlen;
break;
#else
/*
Because of a bug in MySQL 4.1 where length was extended for ENUM and SET
fields for every ALTER TABLE, we have to recalculate lengths here
*/
case MYSQL_TYPE_ENUM:
{ {
case MYSQL_TYPE_TINY_BLOB: uint32 tot_length, max_length;
case MYSQL_TYPE_MEDIUM_BLOB: calculate_interval_lengths(current_thd, interval,
case MYSQL_TYPE_LONG_BLOB: &max_length, &tot_length);
case MYSQL_TYPE_BLOB: length= max_length * charset->mbmaxlen;
case MYSQL_TYPE_VAR_STRING: break;
case MYSQL_TYPE_STRING: }
length*= charset->mbmaxlen; case MYSQL_TYPE_SET:
pack_length= calc_pack_length(sql_type == FIELD_TYPE_VAR_STRING ? {
FIELD_TYPE_STRING : sql_type, length); uint32 tot_length, max_length;
break; calculate_interval_lengths(current_thd, interval,
case MYSQL_TYPE_ENUM: &max_length, &tot_length);
case MYSQL_TYPE_SET: length= (tot_length + (interval->count - 1)) * charset->mbmaxlen;
length*= charset->mbmaxlen; break;
break; }
default: #endif
/* do nothing */ default:
break; /* do nothing */
break;
} }
} }
...@@ -6085,6 +6107,8 @@ create_field::create_field(Field *old_field,Field *orig_field) ...@@ -6085,6 +6107,8 @@ create_field::create_field(Field *old_field,Field *orig_field)
} }
length=(length+charset->mbmaxlen-1)/charset->mbmaxlen; // QQ: Probably not needed length=(length+charset->mbmaxlen-1)/charset->mbmaxlen; // QQ: Probably not needed
break; break;
case MYSQL_TYPE_ENUM:
case MYSQL_TYPE_SET:
case FIELD_TYPE_STRING: case FIELD_TYPE_STRING:
case FIELD_TYPE_VAR_STRING: case FIELD_TYPE_VAR_STRING:
length=(length+charset->mbmaxlen-1)/charset->mbmaxlen; length=(length+charset->mbmaxlen-1)/charset->mbmaxlen;
......
...@@ -2365,10 +2365,10 @@ Item_func_regex::fix_fields(THD *thd, TABLE_LIST *tables, Item **ref) ...@@ -2365,10 +2365,10 @@ Item_func_regex::fix_fields(THD *thd, TABLE_LIST *tables, Item **ref)
} }
int error; int error;
if ((error= regcomp(&preg,res->c_ptr(), if ((error= regcomp(&preg,res->c_ptr(),
((cmp_collation.collation->state & MY_CS_BINSORT) || ((cmp_collation.collation->state &
(cmp_collation.collation->state & MY_CS_CSSORT)) ? (MY_CS_BINSORT | MY_CS_CSSORT)) ?
REG_EXTENDED | REG_NOSUB : REG_EXTENDED | REG_NOSUB :
REG_EXTENDED | REG_NOSUB | REG_ICASE, REG_EXTENDED | REG_NOSUB | REG_ICASE),
cmp_collation.collation))) cmp_collation.collation)))
{ {
(void) regerror(error,&preg,buff,sizeof(buff)); (void) regerror(error,&preg,buff,sizeof(buff));
...@@ -2417,10 +2417,10 @@ longlong Item_func_regex::val_int() ...@@ -2417,10 +2417,10 @@ longlong Item_func_regex::val_int()
regex_compiled=0; regex_compiled=0;
} }
if (regcomp(&preg,res2->c_ptr(), if (regcomp(&preg,res2->c_ptr(),
((cmp_collation.collation->state & MY_CS_BINSORT) || ((cmp_collation.collation->state &
(cmp_collation.collation->state & MY_CS_CSSORT)) ? (MY_CS_BINSORT | MY_CS_CSSORT)) ?
REG_EXTENDED | REG_NOSUB : REG_EXTENDED | REG_NOSUB :
REG_EXTENDED | REG_NOSUB | REG_ICASE, REG_EXTENDED | REG_NOSUB | REG_ICASE),
cmp_collation.collation)) cmp_collation.collation))
{ {
null_value=1; null_value=1;
......
...@@ -370,6 +370,8 @@ int insert_precheck(THD *thd, TABLE_LIST *tables); ...@@ -370,6 +370,8 @@ int insert_precheck(THD *thd, TABLE_LIST *tables);
int create_table_precheck(THD *thd, TABLE_LIST *tables, int create_table_precheck(THD *thd, TABLE_LIST *tables,
TABLE_LIST *create_table); TABLE_LIST *create_table);
Item *negate_expression(THD *thd, Item *expr); Item *negate_expression(THD *thd, Item *expr);
void calculate_interval_lengths(THD *thd, TYPELIB *interval,
uint *max_length, uint *tot_length);
#include "sql_class.h" #include "sql_class.h"
#include "opt_range.h" #include "opt_range.h"
......
...@@ -2725,24 +2725,23 @@ sys_var *find_sys_var(const char *str, uint length) ...@@ -2725,24 +2725,23 @@ sys_var *find_sys_var(const char *str, uint length)
int sql_set_variables(THD *thd, List<set_var_base> *var_list) int sql_set_variables(THD *thd, List<set_var_base> *var_list)
{ {
int error= 0; int error;
List_iterator_fast<set_var_base> it(*var_list); List_iterator_fast<set_var_base> it(*var_list);
DBUG_ENTER("sql_set_variables"); DBUG_ENTER("sql_set_variables");
set_var_base *var; set_var_base *var;
while ((var=it++)) while ((var=it++))
{ {
if ((error=var->check(thd))) if ((error= var->check(thd)))
goto err; goto err;
} }
if (!thd->net.report_error) if (!(error= test(thd->net.report_error)))
{ {
it.rewind(); it.rewind();
while ((var= it++)) while ((var= it++))
error|= var->update(thd); // Returns 0, -1 or 1 error|= var->update(thd); // Returns 0, -1 or 1
} }
else
error= 1;
err: err:
free_underlaid_joins(thd, &thd->lex->select_lex); free_underlaid_joins(thd, &thd->lex->select_lex);
DBUG_RETURN(error); DBUG_RETURN(error);
......
...@@ -814,8 +814,7 @@ int mysql_prepare_table(THD *thd, HA_CREATE_INFO *create_info, ...@@ -814,8 +814,7 @@ int mysql_prepare_table(THD *thd, HA_CREATE_INFO *create_info,
DBUG_RETURN(-1); DBUG_RETURN(-1);
} }
} }
else else if (key_info->algorithm == HA_KEY_ALG_RTREE)
if (key_info->algorithm == HA_KEY_ALG_RTREE)
{ {
#ifdef HAVE_RTREE_KEYS #ifdef HAVE_RTREE_KEYS
if ((key_info->key_parts & 1) == 1) if ((key_info->key_parts & 1) == 1)
...@@ -839,6 +838,8 @@ int mysql_prepare_table(THD *thd, HA_CREATE_INFO *create_info, ...@@ -839,6 +838,8 @@ int mysql_prepare_table(THD *thd, HA_CREATE_INFO *create_info,
CHARSET_INFO *ft_key_charset=0; // for FULLTEXT CHARSET_INFO *ft_key_charset=0; // for FULLTEXT
for (uint column_nr=0 ; (column=cols++) ; column_nr++) for (uint column_nr=0 ; (column=cols++) ; column_nr++)
{ {
key_part_spec *dup_column;
it.rewind(); it.rewind();
field=0; field=0;
while ((sql_field=it++) && while ((sql_field=it++) &&
...@@ -853,9 +854,8 @@ int mysql_prepare_table(THD *thd, HA_CREATE_INFO *create_info, ...@@ -853,9 +854,8 @@ int mysql_prepare_table(THD *thd, HA_CREATE_INFO *create_info,
column->field_name); column->field_name);
DBUG_RETURN(-1); DBUG_RETURN(-1);
} }
for (uint dup_nr= 0; dup_nr < column_nr; dup_nr++) while ((dup_column= cols2++) != column)
{ {
key_part_spec *dup_column= cols2++;
if (!my_strcasecmp(system_charset_info, if (!my_strcasecmp(system_charset_info,
column->field_name, dup_column->field_name)) column->field_name, dup_column->field_name))
{ {
...@@ -866,12 +866,6 @@ int mysql_prepare_table(THD *thd, HA_CREATE_INFO *create_info, ...@@ -866,12 +866,6 @@ int mysql_prepare_table(THD *thd, HA_CREATE_INFO *create_info,
} }
} }
cols2.rewind(); cols2.rewind();
/* for fulltext keys keyseg length is 1 for blobs (it's ignored in
ft code anyway, and 0 (set to column width later) for char's.
it has to be correct col width for char's, as char data are not
prefixed with length (unlike blobs, where ft code takes data length
from a data prefix, ignoring column->length).
*/
if (key->type == Key::FULLTEXT) if (key->type == Key::FULLTEXT)
{ {
if ((sql_field->sql_type != FIELD_TYPE_STRING && if ((sql_field->sql_type != FIELD_TYPE_STRING &&
......
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