Commit a5d7e701 authored by Sergei Golubchik's avatar Sergei Golubchik

MDEV-5029 Crash in MariaDB 5.5.33 with .frm from older MariaDB release

don't set TABLE_SHARE::keys before TABLE_SHARE::key_info is set,
otherwise an error might leave only the first property set and it will
confuse TABLE_SHARE::destroy()
parent 68df7a01
show create table t1;
ERROR 42000: Unknown storage engine 'InnoDB'
#
# MDEV-5029 Crash in MariaDB 5.5.33 with .frm from older MariaDB release
#
# a.k.a. fail to open an frm with indexes:
let $datadir=`select @@datadir`;
copy_file std_data/mdev5029.frm $datadir/test/t1.frm;
--error ER_UNKNOWN_STORAGE_ENGINE
show create table t1;
remove_file $datadir/test/t1.frm;
...@@ -890,6 +890,9 @@ static bool create_key_infos(uchar *strpos, uint keys, KEY *keyinfo, uint new_fr ...@@ -890,6 +890,9 @@ static bool create_key_infos(uchar *strpos, uint keys, KEY *keyinfo, uint new_fr
DBUG_ASSERT(test(keyinfo->flags & HA_USES_COMMENT) == DBUG_ASSERT(test(keyinfo->flags & HA_USES_COMMENT) ==
(keyinfo->comment.length > 0)); (keyinfo->comment.length > 0));
} }
share->keys= keys; // do it *after* all key_info's are initialized
return 0; return 0;
} }
...@@ -1020,12 +1023,12 @@ static int open_binary_frm(THD *thd, TABLE_SHARE *share, uchar *head, ...@@ -1020,12 +1023,12 @@ static int open_binary_frm(THD *thd, TABLE_SHARE *share, uchar *head,
goto err; /* purecov: inspected */ goto err; /* purecov: inspected */
if (disk_buff[0] & 0x80) if (disk_buff[0] & 0x80)
{ {
share->keys= keys= (disk_buff[1] << 7) | (disk_buff[0] & 0x7f); keys= (disk_buff[1] << 7) | (disk_buff[0] & 0x7f);
share->key_parts= key_parts= uint2korr(disk_buff+2); share->key_parts= key_parts= uint2korr(disk_buff+2);
} }
else else
{ {
share->keys= keys= disk_buff[0]; keys= disk_buff[0];
share->key_parts= key_parts= disk_buff[1]; share->key_parts= key_parts= disk_buff[1];
} }
share->keys_for_keyread.init(0); share->keys_for_keyread.init(0);
...@@ -1283,7 +1286,7 @@ static int open_binary_frm(THD *thd, TABLE_SHARE *share, uchar *head, ...@@ -1283,7 +1286,7 @@ static int open_binary_frm(THD *thd, TABLE_SHARE *share, uchar *head,
share->comment.length); share->comment.length);
} }
DBUG_PRINT("info",("i_count: %d i_parts: %d index: %d n_length: %d int_length: %d com_length: %d vcol_screen_length: %d", interval_count,interval_parts, share->keys,n_length,int_length, com_length, vcol_screen_length)); DBUG_PRINT("info",("i_count: %d i_parts: %d index: %d n_length: %d int_length: %d com_length: %d vcol_screen_length: %d", interval_count,interval_parts, keys,n_length,int_length, com_length, vcol_screen_length));
if (!(field_ptr = (Field **) if (!(field_ptr = (Field **)
...@@ -1671,7 +1674,7 @@ static int open_binary_frm(THD *thd, TABLE_SHARE *share, uchar *head, ...@@ -1671,7 +1674,7 @@ static int open_binary_frm(THD *thd, TABLE_SHARE *share, uchar *head,
} }
} }
for (uint key=0 ; key < share->keys ; key++,keyinfo++) for (uint key=0 ; key < keys ; key++,keyinfo++)
{ {
uint usable_parts= 0; uint usable_parts= 0;
keyinfo->name=(char*) share->keynames.type_names[key]; keyinfo->name=(char*) share->keynames.type_names[key];
...@@ -1945,7 +1948,7 @@ static int open_binary_frm(THD *thd, TABLE_SHARE *share, uchar *head, ...@@ -1945,7 +1948,7 @@ static int open_binary_frm(THD *thd, TABLE_SHARE *share, uchar *head,
{ {
reg_field= *share->found_next_number_field; reg_field= *share->found_next_number_field;
if ((int) (share->next_number_index= (uint) if ((int) (share->next_number_index= (uint)
find_ref_key(share->key_info, share->keys, find_ref_key(share->key_info, keys,
share->default_values, reg_field, share->default_values, reg_field,
&share->next_number_key_offset, &share->next_number_key_offset,
&share->next_number_keypart)) < 0) &share->next_number_keypart)) < 0)
......
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