1
/* -*- mode: c++; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
* vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
4
* Copyright (C) 2009 Sun Microsystems
6
* This program is free software; you can redistribute it and/or modify
7
* it under the terms of the GNU General Public License as published by
8
* the Free Software Foundation; either version 2 of the License, or
9
* (at your option) any later version.
11
* This program is distributed in the hope that it will be useful,
12
* but WITHOUT ANY WARRANTY; without even the implied warranty of
13
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14
* GNU General Public License for more details.
16
* You should have received a copy of the GNU General Public License
17
* along with this program; if not, write to the Free Software
18
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
27
#include "drizzled/show.h"
28
#include "drizzled/lock.h"
29
#include "drizzled/session.h"
30
#include "drizzled/statement/alter_table.h"
31
#include "drizzled/global_charset_info.h"
34
#include "drizzled/gettext.h"
35
#include "drizzled/data_home.h"
36
#include "drizzled/sql_table.h"
37
#include "drizzled/table_proto.h"
38
#include "drizzled/optimizer/range.h"
39
#include "drizzled/time_functions.h"
40
#include "drizzled/records.h"
41
#include "drizzled/pthread_globals.h"
42
#include "drizzled/internal/my_sys.h"
43
#include "drizzled/internal/iocache.h"
45
#include "drizzled/transaction_services.h"
52
extern pid_t current_pid;
54
static int copy_data_between_tables(Session *session,
55
Table *from,Table *to,
56
List<CreateField> &create,
62
enum enum_enable_or_disable keys_onoff,
63
bool error_if_not_empty);
65
static bool mysql_prepare_alter_table(Session *session,
67
HA_CREATE_INFO *create_info,
68
const message::Table &original_proto,
69
message::Table &table_message,
70
AlterInfo *alter_info);
72
static int create_temporary_table(Session *session,
73
TableIdentifier &identifier,
74
HA_CREATE_INFO *create_info,
75
message::Table &create_message,
76
AlterInfo *alter_info);
78
static Table *open_alter_table(Session *session, Table *table, TableIdentifier &identifier);
80
bool statement::AlterTable::execute()
82
TableList *first_table= (TableList *) session->lex->select_lex.table_list.first;
83
TableList *all_tables= session->lex->query_tables;
84
assert(first_table == all_tables && first_table != 0);
85
Select_Lex *select_lex= &session->lex->select_lex;
86
bool need_start_waiting= false;
91
plugin::StorageEngine::findByName(*session, create_table_message.engine().name());
93
if (create_info.db_type == NULL)
95
my_error(ER_UNKNOWN_STORAGE_ENGINE, MYF(0),
96
create_table_message.engine().name().c_str());
102
/* Must be set in the parser */
103
assert(select_lex->db);
105
/* Chicken/Egg... we need to search for the table, to know if the table exists, so we can build a full identifier from it */
106
message::Table original_table_message;
108
TableIdentifier identifier(first_table->db, first_table->table_name);
109
if (plugin::StorageEngine::getTableDefinition(*session, identifier, original_table_message) != EEXIST)
111
my_error(ER_BAD_TABLE_ERROR, MYF(0), identifier.getSQLPath().c_str());
115
if (not create_info.db_type)
118
plugin::StorageEngine::findByName(*session, original_table_message.engine().name());
120
if (not create_info.db_type)
122
my_error(ER_BAD_TABLE_ERROR, MYF(0), identifier.getSQLPath().c_str());
128
if (not validateCreateTableOption())
133
/* ALTER TABLE ends previous transaction */
134
if (not session->endActiveTransaction())
139
if (not (need_start_waiting= ! wait_if_global_read_lock(session, 0, 1)))
145
if (original_table_message.type() == message::Table::STANDARD )
147
TableIdentifier identifier(first_table->db, first_table->table_name);
148
TableIdentifier new_identifier(select_lex->db ? select_lex->db : first_table->db,
149
session->lex->name.str ? session->lex->name.str : first_table->table_name);
151
res= alter_table(session,
155
original_table_message,
156
create_table_message,
159
select_lex->order_list.elements,
160
(order_st *) select_lex->order_list.first,
161
session->lex->ignore);
165
Table *table= session->find_temporary_table(first_table);
168
TableIdentifier identifier(first_table->db, first_table->table_name, table->getMutableShare()->getPath());
169
TableIdentifier new_identifier(select_lex->db ? select_lex->db : first_table->db,
170
session->lex->name.str ? session->lex->name.str : first_table->table_name,
171
table->getMutableShare()->getPath());
173
res= alter_table(session,
177
original_table_message,
178
create_table_message,
181
select_lex->order_list.elements,
182
(order_st *) select_lex->order_list.first,
183
session->lex->ignore);
188
Release the protection against the global read lock and wake
189
everyone, who might want to set a global read lock.
191
start_waiting_global_read_lock(session);
197
Prepare column and key definitions for CREATE TABLE in ALTER Table.
199
This function transforms parse output of ALTER Table - lists of
200
columns and keys to add, drop or modify into, essentially,
201
CREATE TABLE definition - a list of columns and keys of the new
202
table. While doing so, it also performs some (bug not all)
205
This function is invoked when we know that we're going to
206
perform ALTER Table via a temporary table -- i.e. fast ALTER Table
207
is not possible, perhaps because the ALTER statement contains
208
instructions that require change in table data, not only in
209
table definition or indexes.
211
@param[in,out] session thread handle. Used as a memory pool
212
and source of environment information.
213
@param[in] table the source table, open and locked
214
Used as an interface to the storage engine
215
to acquire additional information about
217
@param[in,out] create_info A blob with CREATE/ALTER Table
219
@param[in,out] alter_info Another blob with ALTER/CREATE parameters.
220
Originally create_info was used only in
221
CREATE TABLE and alter_info only in ALTER Table.
222
But since ALTER might end-up doing CREATE,
223
this distinction is gone and we just carry
224
around two structures.
227
Fills various create_info members based on information retrieved
228
from the storage engine.
229
Sets create_info->varchar if the table has a VARCHAR column.
230
Prepares alter_info->create_list and alter_info->key_list with
231
columns and keys of the new table.
232
@retval true error, out of memory or a semantical error in ALTER
234
@retval false success
236
static bool mysql_prepare_alter_table(Session *session,
238
HA_CREATE_INFO *create_info,
239
const message::Table &original_proto,
240
message::Table &table_message,
241
AlterInfo *alter_info)
243
/* New column definitions are added here */
244
List<CreateField> new_create_list;
245
/* New key definitions are added here */
246
List<Key> new_key_list;
247
List_iterator<AlterDrop> drop_it(alter_info->drop_list);
248
List_iterator<CreateField> def_it(alter_info->create_list);
249
List_iterator<AlterColumn> alter_it(alter_info->alter_list);
250
List_iterator<Key> key_it(alter_info->key_list);
251
List_iterator<CreateField> find_it(new_create_list);
252
List_iterator<CreateField> field_it(new_create_list);
253
List<Key_part_spec> key_parts;
254
uint32_t used_fields= create_info->used_fields;
255
KeyInfo *key_info= table->key_info;
258
/* Let new create options override the old ones */
259
message::Table::TableOptions *table_options;
260
table_options= table_message.mutable_options();
262
if (! (used_fields & HA_CREATE_USED_DEFAULT_CHARSET))
263
create_info->default_table_charset= table->getShare()->table_charset;
264
if (! (used_fields & HA_CREATE_USED_AUTO) &&
265
table->found_next_number_field)
267
/* Table has an autoincrement, copy value to new table */
268
table->cursor->info(HA_STATUS_AUTO);
269
create_info->auto_increment_value= table->cursor->stats.auto_increment_value;
270
if (create_info->auto_increment_value != original_proto.options().auto_increment_value())
271
table_options->set_has_user_set_auto_increment_value(false);
273
table->restoreRecordAsDefault(); /* Empty record for DEFAULT */
276
/* First collect all fields from table which isn't in drop_list */
279
for (f_ptr= table->getFields(); (field= *f_ptr); f_ptr++)
281
/* Check if field should be dropped */
284
while ((drop= drop_it++))
286
if (drop->type == AlterDrop::COLUMN &&
287
! my_strcasecmp(system_charset_info, field->field_name, drop->name))
289
/* Reset auto_increment value if it was dropped */
290
if (MTYP_TYPENR(field->unireg_check) == Field::NEXT_NUMBER &&
291
! (used_fields & HA_CREATE_USED_AUTO))
293
create_info->auto_increment_value= 0;
294
create_info->used_fields|= HA_CREATE_USED_AUTO;
305
/* Mark that we will read the field */
308
/* Check if field is changed */
310
while ((def= def_it++))
313
! my_strcasecmp(system_charset_info, field->field_name, def->change))
318
/* Field is changed */
322
new_create_list.push_back(def);
329
This field was not dropped and not changed, add it to the list
332
def= new CreateField(field, field);
333
new_create_list.push_back(def);
334
alter_it.rewind(); /* Change default if ALTER */
336
while ((alter= alter_it++))
338
if (! my_strcasecmp(system_charset_info,field->field_name, alter->name))
343
if (def->sql_type == DRIZZLE_TYPE_BLOB)
345
my_error(ER_BLOB_CANT_HAVE_DEFAULT, MYF(0), def->change);
348
if ((def->def= alter->def))
350
/* Use new default */
351
def->flags&= ~NO_DEFAULT_VALUE_FLAG;
354
def->flags|= NO_DEFAULT_VALUE_FLAG;
360
while ((def= def_it++)) /* Add new columns */
362
if (def->change && ! def->field)
364
my_error(ER_BAD_FIELD_ERROR, MYF(0), def->change, table->getMutableShare()->getTableName());
368
Check that the DATE/DATETIME not null field we are going to add is
369
either has a default value or the '0000-00-00' is allowed by the
371
If the '0000-00-00' value isn't allowed then raise the error_if_not_empty
372
flag to allow ALTER Table only if the table to be altered is empty.
374
if ((def->sql_type == DRIZZLE_TYPE_DATE ||
375
def->sql_type == DRIZZLE_TYPE_DATETIME) &&
376
! alter_info->datetime_field &&
377
! (~def->flags & (NO_DEFAULT_VALUE_FLAG | NOT_NULL_FLAG)))
379
alter_info->datetime_field= def;
380
alter_info->error_if_not_empty= true;
383
new_create_list.push_back(def);
384
else if (def->after == first_keyword)
385
new_create_list.push_front(def);
390
while ((find= find_it++)) /* Add new columns */
392
if (! my_strcasecmp(system_charset_info,def->after, find->field_name))
397
my_error(ER_BAD_FIELD_ERROR, MYF(0), def->after, table->getMutableShare()->getTableName());
400
find_it.after(def); /* Put element after this */
402
XXX: hack for Bug#28427.
403
If column order has changed, force OFFLINE ALTER Table
404
without querying engine capabilities. If we ever have an
405
engine that supports online ALTER Table CHANGE COLUMN
406
<name> AFTER <name1> (Falcon?), this fix will effectively
407
disable the capability.
408
TODO: detect the situation in compare_tables, behave based
409
on engine capabilities.
411
if (alter_info->build_method == HA_BUILD_ONLINE)
413
my_error(ER_NOT_SUPPORTED_YET, MYF(0), session->query.c_str());
416
alter_info->build_method= HA_BUILD_OFFLINE;
419
if (alter_info->alter_list.elements)
421
my_error(ER_BAD_FIELD_ERROR,
423
alter_info->alter_list.head()->name,
424
table->getMutableShare()->getTableName());
427
if (! new_create_list.elements)
429
my_message(ER_CANT_REMOVE_ALL_FIELDS,
430
ER(ER_CANT_REMOVE_ALL_FIELDS),
436
Collect all keys which isn't in drop list. Add only those
437
for which some fields exists.
439
for (uint32_t i= 0; i < table->getShare()->sizeKeys(); i++, key_info++)
441
char *key_name= key_info->name;
444
while ((drop= drop_it++))
446
if (drop->type == AlterDrop::KEY &&
447
! my_strcasecmp(system_charset_info, key_name, drop->name))
456
KeyPartInfo *key_part= key_info->key_part;
458
for (uint32_t j= 0; j < key_info->key_parts; j++, key_part++)
460
if (! key_part->field)
461
continue; /* Wrong field (from UNIREG) */
463
const char *key_part_name= key_part->field->field_name;
466
while ((cfield= field_it++))
470
if (! my_strcasecmp(system_charset_info, key_part_name, cfield->change))
473
else if (! my_strcasecmp(system_charset_info, key_part_name, cfield->field_name))
477
continue; /* Field is removed */
479
uint32_t key_part_length= key_part->length;
480
if (cfield->field) /* Not new field */
483
If the field can't have only a part used in a key according to its
484
new type, or should not be used partially according to its
485
previous type, or the field length is less than the key part
486
length, unset the key part length.
488
We also unset the key part length if it is the same as the
489
old field's length, so the whole new field will be used.
491
BLOBs may have cfield->length == 0, which is why we test it before
492
checking whether cfield->length < key_part_length (in chars).
494
if (! Field::type_can_have_key_part(cfield->field->type()) ||
495
! Field::type_can_have_key_part(cfield->sql_type) ||
496
(cfield->field->field_length == key_part_length) ||
498
(cfield->length < key_part_length / key_part->field->charset()->mbmaxlen)))
499
key_part_length= 0; /* Use whole field */
501
key_part_length/= key_part->field->charset()->mbmaxlen;
502
key_parts.push_back(new Key_part_spec(cfield->field_name,
503
strlen(cfield->field_name),
506
if (key_parts.elements)
508
KEY_CREATE_INFO key_create_info;
510
enum Key::Keytype key_type;
511
memset(&key_create_info, 0, sizeof(key_create_info));
513
key_create_info.algorithm= key_info->algorithm;
514
if (key_info->flags & HA_USES_BLOCK_SIZE)
515
key_create_info.block_size= key_info->block_size;
516
if (key_info->flags & HA_USES_COMMENT)
517
key_create_info.comment= key_info->comment;
519
if (key_info->flags & HA_NOSAME)
521
if (is_primary_key_name(key_name))
522
key_type= Key::PRIMARY;
524
key_type= Key::UNIQUE;
527
key_type= Key::MULTIPLE;
529
key= new Key(key_type,
533
test(key_info->flags & HA_GENERATED_KEY),
535
new_key_list.push_back(key);
539
/* Copy over existing foreign keys */
540
for (int j= 0; j < original_proto.fk_constraint_size(); j++)
544
while ((drop= drop_it++))
546
if (drop->type == AlterDrop::FOREIGN_KEY &&
547
! my_strcasecmp(system_charset_info, original_proto.fk_constraint(j).name().c_str(), drop->name))
558
message::Table::ForeignKeyConstraint *pfkey= table_message.add_fk_constraint();
559
*pfkey= original_proto.fk_constraint(j);
564
while ((key= key_it++)) /* Add new keys */
566
if (key->type == Key::FOREIGN_KEY)
568
if (((Foreign_key *)key)->validate(new_create_list))
571
Foreign_key *fkey= (Foreign_key*)key;
572
add_foreign_key_to_table_message(&table_message,
582
if (key->type != Key::FOREIGN_KEY)
583
new_key_list.push_back(key);
585
if (key->name.str && is_primary_key_name(key->name.str))
587
my_error(ER_WRONG_NAME_FOR_INDEX,
595
/* Fix names of foreign keys being added */
596
for (int j= 0; j < table_message.fk_constraint_size(); j++)
598
if (! table_message.fk_constraint(j).has_name())
600
std::string name(table->getMutableShare()->getTableName());
603
name.append("_ibfk_");
604
snprintf(number, sizeof(number), "%d", j+1);
607
message::Table::ForeignKeyConstraint *pfkey= table_message.mutable_fk_constraint(j);
608
pfkey->set_name(name);
612
if (alter_info->drop_list.elements)
614
my_error(ER_CANT_DROP_FIELD_OR_KEY,
616
alter_info->drop_list.head()->name);
619
if (alter_info->alter_list.elements)
621
my_error(ER_CANT_DROP_FIELD_OR_KEY,
623
alter_info->alter_list.head()->name);
627
if (not table_message.options().has_comment()
628
&& table->getMutableShare()->hasComment())
629
table_options->set_comment(table->getMutableShare()->getComment());
631
if (table->getShare()->getType())
633
table_message.set_type(message::Table::TEMPORARY);
636
table_message.set_creation_timestamp(table->getShare()->getTableProto()->creation_timestamp());
638
table_message.set_update_timestamp(time(NULL));
641
alter_info->create_list.swap(new_create_list);
642
alter_info->key_list.swap(new_key_list);
645
size_t num_engine_options= table_message.engine().options_size();
646
size_t original_num_engine_options= original_proto.engine().options_size();
647
for (size_t x= 0; x < original_num_engine_options; ++x)
651
for (size_t y= 0; y < num_engine_options; ++y)
653
found= not table_message.engine().options(y).name().compare(original_proto.engine().options(x).name());
661
message::Engine::Option *opt= table_message.mutable_engine()->add_options();
663
opt->set_name(original_proto.engine().options(x).name());
664
opt->set_state(original_proto.engine().options(x).state());
671
/* table_list should contain just one table */
672
static int mysql_discard_or_import_tablespace(Session *session,
673
TableList *table_list,
674
enum tablespace_op_type tablespace_op)
681
Note that DISCARD/IMPORT TABLESPACE always is the only operation in an
685
TransactionServices &transaction_services= TransactionServices::singleton();
686
session->set_proc_info("discard_or_import_tablespace");
688
discard= test(tablespace_op == DISCARD_TABLESPACE);
691
We set this flag so that ha_innobase::open and ::external_lock() do
692
not complain when we lock the table
694
session->tablespace_op= true;
695
if (!(table= session->openTableLock(table_list, TL_WRITE)))
697
session->tablespace_op= false;
701
error= table->cursor->ha_discard_or_import_tablespace(discard);
703
session->set_proc_info("end");
708
/* The ALTER Table is always in its own transaction */
709
error= transaction_services.autocommitOrRollback(session, false);
710
if (! session->endActiveTransaction())
714
write_bin_log(session, session->query.c_str());
717
(void) transaction_services.autocommitOrRollback(session, error);
718
session->tablespace_op=false;
726
table->print_error(error, MYF(0));
732
Manages enabling/disabling of indexes for ALTER Table
735
alter_table_manage_keys()
737
indexes_were_disabled Whether the indexes of the from table
739
keys_onoff ENABLE | DISABLE | LEAVE_AS_IS
745
static bool alter_table_manage_keys(Session *session,
746
Table *table, int indexes_were_disabled,
747
enum enum_enable_or_disable keys_onoff)
750
switch (keys_onoff) {
752
error= table->cursor->ha_enable_indexes(HA_KEY_SWITCH_NONUNIQ_SAVE);
755
if (not indexes_were_disabled)
757
/* fall-through: disabled indexes */
759
error= table->cursor->ha_disable_indexes(HA_KEY_SWITCH_NONUNIQ_SAVE);
762
if (error == HA_ERR_WRONG_COMMAND)
764
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
765
ER_ILLEGAL_HA, ER(ER_ILLEGAL_HA),
766
table->getMutableShare()->getTableName());
769
table->print_error(error, MYF(0));
774
static bool lockTableIfDifferent(Session &session,
775
TableIdentifier &original_table_identifier,
776
TableIdentifier &new_table_identifier,
779
/* Check that we are not trying to rename to an existing table */
780
if (not (original_table_identifier == new_table_identifier))
782
if (original_table_identifier.isTmp())
785
if (session.find_temporary_table(new_table_identifier))
787
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), new_table_identifier.getSQLPath().c_str());
793
if (session.lock_table_name_if_not_cached(new_table_identifier, &name_lock))
800
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), new_table_identifier.getSQLPath().c_str());
804
if (plugin::StorageEngine::doesTableExist(session, new_table_identifier))
806
/* Table will be closed by Session::executeCommand() */
807
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), new_table_identifier.getSQLPath().c_str());
809
LOCK_open.lock(); /* ALTER TABLe */
810
session.unlink_open_table(name_lock);
826
session Thread handle
827
new_db If there is a RENAME clause
828
new_name If there is a RENAME clause
829
create_info Information from the parsing phase about new
831
table_list The table to change.
832
alter_info Lists of fields, keys to be changed, added
834
order_num How many ORDER BY fields has been specified.
835
order List of fields to order_st BY.
836
ignore Whether we have ALTER IGNORE Table
839
This is a veery long function and is everything but the kitchen sink :)
840
It is used to alter a table and not only by ALTER Table but also
841
CREATE|DROP INDEX are mapped on this function.
843
When the ALTER Table statement just does a RENAME or ENABLE|DISABLE KEYS,
844
or both, then this function short cuts its operation by renaming
845
the table and/or enabling/disabling the keys. In this case, the FRM is
846
not changed, directly by alter_table. However, if there is a
847
RENAME + change of a field, or an index, the short cut is not used.
848
See how `create_list` is used to generate the new FRM regarding the
849
structure of the fields. The same is done for the indices of the table.
851
Important is the fact, that this function tries to do as little work as
852
possible, by finding out whether a intermediate table is needed to copy
853
data into and when finishing the altering to use it as the original table.
854
For this reason the function compare_tables() is called, which decides
855
based on all kind of data how similar are the new and the original
863
static bool internal_alter_table(Session *session,
865
TableIdentifier &original_table_identifier,
866
TableIdentifier &new_table_identifier,
867
HA_CREATE_INFO *create_info,
868
const message::Table &original_proto,
869
message::Table &create_proto,
870
TableList *table_list,
871
AlterInfo *alter_info,
882
session->set_proc_info("init");
884
table->use_all_columns();
886
plugin::StorageEngine *new_engine;
887
plugin::StorageEngine *original_engine;
889
original_engine= table->getMutableShare()->getEngine();
891
if (not create_info->db_type)
893
create_info->db_type= original_engine;
895
new_engine= create_info->db_type;
898
create_proto.set_schema(new_table_identifier.getSchemaName());
899
create_proto.set_type(new_table_identifier.getType());
902
@todo Have a check on the table definition for FK in the future
903
to remove the need for the cursor. (aka can_switch_engines())
905
if (new_engine != original_engine &&
906
not table->cursor->can_switch_engines())
909
my_error(ER_ROW_IS_REFERENCED, MYF(0));
914
if (original_engine->check_flag(HTON_BIT_ALTER_NOT_SUPPORTED) ||
915
new_engine->check_flag(HTON_BIT_ALTER_NOT_SUPPORTED))
917
my_error(ER_ILLEGAL_HA, MYF(0), new_table_identifier.getSQLPath().c_str());
922
session->set_proc_info("setup");
925
* test if no other bits except ALTER_RENAME and ALTER_KEYS_ONOFF are set
931
tmp.reset(ALTER_RENAME);
932
tmp.reset(ALTER_KEYS_ONOFF);
933
tmp&= alter_info->flags;
935
if (! (tmp.any()) && ! table->getShare()->getType()) // no need to touch frm
937
switch (alter_info->keys_onoff)
943
wait_while_table_is_used() ensures that table being altered is
944
opened only by this thread and that Table::TableShare::version
945
of Table object corresponding to this table is 0.
946
The latter guarantees that no DML statement will open this table
947
until ALTER Table finishes (i.e. until close_thread_tables())
948
while the fact that the table is still open gives us protection
949
from concurrent DDL statements.
951
LOCK_open.lock(); /* DDL wait for/blocker */
952
wait_while_table_is_used(session, table, HA_EXTRA_FORCE_REOPEN);
954
error= table->cursor->ha_enable_indexes(HA_KEY_SWITCH_NONUNIQ_SAVE);
955
/* COND_refresh will be signaled in close_thread_tables() */
958
LOCK_open.lock(); /* DDL wait for/blocker */
959
wait_while_table_is_used(session, table, HA_EXTRA_FORCE_REOPEN);
961
error=table->cursor->ha_disable_indexes(HA_KEY_SWITCH_NONUNIQ_SAVE);
962
/* COND_refresh will be signaled in close_thread_tables() */
970
if (error == HA_ERR_WRONG_COMMAND)
973
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
974
ER_ILLEGAL_HA, ER(ER_ILLEGAL_HA),
978
LOCK_open.lock(); /* Lock to remove all instances of table from table cache before ALTER */
980
Unlike to the above case close_cached_table() below will remove ALL
981
instances of Table from table cache (it will also remove table lock
982
held by this thread). So to make actual table renaming and writing
983
to binlog atomic we have to put them into the same critical section
984
protected by LOCK_open mutex. This also removes gap for races between
985
access() and mysql_rename_table() calls.
988
if (error == 0 && not (original_table_identifier == new_table_identifier))
990
session->set_proc_info("rename");
992
Then do a 'simple' rename of the table. First we need to close all
993
instances of 'source' table.
995
session->close_cached_table(table);
997
Then, we want check once again that target table does not exist.
998
Actually the order of these two steps does not matter since
999
earlier we took name-lock on the target table, so we do them
1000
in this particular order only to be consistent with 5.0, in which
1001
we don't take this name-lock and where this order really matters.
1002
@todo Investigate if we need this access() check at all.
1004
if (plugin::StorageEngine::doesTableExist(*session, new_table_identifier))
1006
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), new_table_identifier.getSQLPath().c_str());
1011
if (mysql_rename_table(*session, original_engine, original_table_identifier, new_table_identifier))
1018
if (error == HA_ERR_WRONG_COMMAND)
1021
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1022
ER_ILLEGAL_HA, ER(ER_ILLEGAL_HA),
1028
write_bin_log(session, session->query.c_str());
1033
table->print_error(error, MYF(0));
1038
table_list->table= NULL;
1044
/* We have to do full alter table. */
1045
new_engine= create_info->db_type;
1047
if (mysql_prepare_alter_table(session, table, create_info, original_proto, create_proto, alter_info))
1052
set_table_default_charset(create_info, new_table_identifier.getSchemaName().c_str());
1054
alter_info->build_method= HA_BUILD_OFFLINE;
1056
snprintf(tmp_name, sizeof(tmp_name), "%s-%lx_%"PRIx64, TMP_FILE_PREFIX, (unsigned long) current_pid, session->thread_id);
1058
/* Create a temporary table with the new format */
1060
@note we make an internal temporary table unless the table is a temporary table. In last
1061
case we just use it as is. Neither of these tables require locks in order to be
1064
TableIdentifier new_table_as_temporary(original_table_identifier.getSchemaName(),
1066
create_proto.type() != message::Table::TEMPORARY ? message::Table::INTERNAL :
1067
message::Table::TEMPORARY);
1069
error= create_temporary_table(session, new_table_as_temporary, create_info, create_proto, alter_info);
1076
/* Open the table so we need to copy the data to it. */
1077
Table *new_table= open_alter_table(session, table, new_table_as_temporary);
1082
quick_rm_table(*session, new_table_as_temporary);
1086
/* Copy the data if necessary. */
1088
/* We must not ignore bad input! */
1089
session->count_cuted_fields= CHECK_FIELD_ERROR_FOR_NULL; // calc cuted fields
1090
session->cuted_fields= 0L;
1091
session->set_proc_info("copy to tmp table");
1094
/* We don't want update TIMESTAMP fields during ALTER Table. */
1095
new_table->timestamp_field_type= TIMESTAMP_NO_AUTO_SET;
1096
new_table->next_number_field= new_table->found_next_number_field;
1097
error= copy_data_between_tables(session,
1100
alter_info->create_list,
1106
alter_info->keys_onoff,
1107
alter_info->error_if_not_empty);
1109
/* We must not ignore bad input! */
1110
assert(session->count_cuted_fields == CHECK_FIELD_ERROR_FOR_NULL);
1113
/* Now we need to resolve what just happened with the data copy. */
1119
No default value was provided for a DATE/DATETIME field, the
1120
current sql_mode doesn't allow the '0000-00-00' value and
1121
the table to be altered isn't empty.
1124
if (alter_info->error_if_not_empty && session->row_count)
1126
const char *f_val= 0;
1127
enum enum_drizzle_timestamp_type t_type= DRIZZLE_TIMESTAMP_DATE;
1129
switch (alter_info->datetime_field->sql_type)
1131
case DRIZZLE_TYPE_DATE:
1132
f_val= "0000-00-00";
1133
t_type= DRIZZLE_TIMESTAMP_DATE;
1135
case DRIZZLE_TYPE_DATETIME:
1136
f_val= "0000-00-00 00:00:00";
1137
t_type= DRIZZLE_TIMESTAMP_DATETIME;
1140
/* Shouldn't get here. */
1143
bool save_abort_on_warning= session->abort_on_warning;
1144
session->abort_on_warning= true;
1145
make_truncated_value_warning(session, DRIZZLE_ERROR::WARN_LEVEL_ERROR,
1146
f_val, internal::strlength(f_val), t_type,
1147
alter_info->datetime_field->field_name);
1148
session->abort_on_warning= save_abort_on_warning;
1151
if (original_table_identifier.isTmp())
1155
/* close_temporary_table() frees the new_table pointer. */
1156
session->close_temporary_table(new_table);
1160
quick_rm_table(*session, new_table_as_temporary);
1170
Close the intermediate table that will be the new table.
1171
Note that MERGE tables do not have their children attached here.
1173
new_table->intern_close_table();
1174
if (new_table->hasShare())
1176
delete new_table->s;
1183
LOCK_open.lock(); /* ALTER TABLE */
1185
quick_rm_table(*session, new_table_as_temporary);
1191
// Temporary table and success
1192
else if (original_table_identifier.isTmp())
1194
/* Close lock if this is a transactional table */
1197
mysql_unlock_tables(session, session->lock);
1201
/* Remove link to old table and rename the new one */
1202
session->close_temporary_table(table);
1204
/* Should pass the 'new_name' as we store table name in the cache */
1205
new_table->getMutableShare()->setIdentifier(new_table_identifier);
1207
new_table_identifier.setPath(new_table_as_temporary.getPath());
1209
if (mysql_rename_table(*session, new_engine, new_table_as_temporary, new_table_identifier) != 0)
1214
// Normal table success
1220
Close the intermediate table that will be the new table.
1221
Note that MERGE tables do not have their children attached here.
1223
new_table->intern_close_table();
1225
if (new_table->hasShare())
1227
delete new_table->s;
1234
LOCK_open.lock(); /* ALTER TABLE */
1237
Data is copied. Now we:
1238
1) Wait until all other threads close old version of table.
1239
2) Close instances of table open by this thread and replace them
1240
with exclusive name-locks.
1241
3) Rename the old table to a temp name, rename the new one to the
1243
4) If we are under LOCK TABLES and don't do ALTER Table ... RENAME
1244
we reopen new version of table.
1245
5) Write statement to the binary log.
1246
6) If we are under LOCK TABLES and do ALTER Table ... RENAME we
1247
remove name-locks from list of open tables and table cache.
1248
7) If we are not not under LOCK TABLES we rely on close_thread_tables()
1249
call to remove name-locks from table cache and list of open table.
1252
session->set_proc_info("rename result table");
1254
snprintf(old_name, sizeof(old_name), "%s2-%lx-%"PRIx64, TMP_FILE_PREFIX, (unsigned long) current_pid, session->thread_id);
1256
my_casedn_str(files_charset_info, old_name);
1258
wait_while_table_is_used(session, table, HA_EXTRA_PREPARE_FOR_RENAME);
1259
session->close_data_files_and_morph_locks(original_table_identifier);
1264
This leads to the storage engine (SE) not being notified for renames in
1265
mysql_rename_table(), because we just juggle with the FRM and nothing
1266
more. If we have an intermediate table, then we notify the SE that
1267
it should become the actual table. Later, we will recycle the old table.
1268
However, in case of ALTER Table RENAME there might be no intermediate
1269
table. This is when the old and new tables are compatible, according to
1270
compare_table(). Then, we need one additional call to
1272
TableIdentifier original_table_to_drop(original_table_identifier.getSchemaName(),
1273
old_name, create_proto.type() != message::Table::TEMPORARY ? message::Table::INTERNAL :
1274
message::Table::TEMPORARY);
1276
if (mysql_rename_table(*session, original_engine, original_table_identifier, original_table_to_drop))
1279
quick_rm_table(*session, new_table_as_temporary);
1283
if (mysql_rename_table(*session, new_engine, new_table_as_temporary, new_table_identifier) != 0)
1285
/* Try to get everything back. */
1288
quick_rm_table(*session, new_table_identifier);
1290
quick_rm_table(*session, new_table_as_temporary);
1292
mysql_rename_table(*session, original_engine, original_table_to_drop, original_table_identifier);
1296
quick_rm_table(*session, original_table_to_drop);
1303
An error happened while we were holding exclusive name-lock on table
1304
being altered. To be safe under LOCK TABLES we should remove placeholders
1305
from list of open tables list and table cache.
1307
session->unlink_open_table(table);
1315
session->set_proc_info("end");
1317
write_bin_log(session, session->query.c_str());
1318
table_list->table= NULL;
1322
* Field::store() may have called my_error(). If this is
1323
* the case, we must not send an ok packet, since
1324
* Diagnostics_area::is_set() will fail an assert.
1326
if (session->is_error())
1328
/* my_error() was called. Return true (which means error...) */
1332
snprintf(tmp_name, sizeof(tmp_name), ER(ER_INSERT_INFO),
1333
(ulong) (copied + deleted), (ulong) deleted,
1334
(ulong) session->cuted_fields);
1335
session->my_ok(copied + deleted, 0, 0L, tmp_name);
1336
session->some_tables_deleted= 0;
1341
bool alter_table(Session *session,
1342
TableIdentifier &original_table_identifier,
1343
TableIdentifier &new_table_identifier,
1344
HA_CREATE_INFO *create_info,
1345
const message::Table &original_proto,
1346
message::Table &create_proto,
1347
TableList *table_list,
1348
AlterInfo *alter_info,
1356
if (alter_info->tablespace_op != NO_TABLESPACE_OP)
1358
/* DISCARD/IMPORT TABLESPACE is always alone in an ALTER Table */
1359
return mysql_discard_or_import_tablespace(session, table_list, alter_info->tablespace_op);
1362
session->set_proc_info("init");
1364
if (not (table= session->openTableLock(table_list, TL_WRITE_ALLOW_READ)))
1367
session->set_proc_info("gained write lock on table");
1370
Check that we are not trying to rename to an existing table,
1371
if one existed we get a lock, if we can't we error.
1374
Table *name_lock= NULL;
1376
if (not lockTableIfDifferent(*session, original_table_identifier, new_table_identifier, name_lock))
1381
error= internal_alter_table(session,
1383
original_table_identifier,
1384
new_table_identifier,
1396
LOCK_open.lock(); /* ALTER TABLe */
1397
session->unlink_open_table(name_lock);
1407
copy_data_between_tables(Session *session,
1408
Table *from, Table *to,
1409
List<CreateField> &create,
1411
uint32_t order_num, order_st *order,
1414
enum enum_enable_or_disable keys_onoff,
1415
bool error_if_not_empty)
1418
CopyField *copy,*copy_end;
1419
ulong found_count,delete_count;
1421
SortField *sortorder;
1425
List<Item> all_fields;
1426
ha_rows examined_rows;
1427
bool auto_increment_field_copied= 0;
1428
uint64_t prev_insert_id;
1431
Turn off recovery logging since rollback of an alter table is to
1432
delete the new table so there is no need to log the changes to it.
1434
This needs to be done before external_lock
1436
TransactionServices &transaction_services= TransactionServices::singleton();
1441
* Since open_temporary_table() doesn't invoke mysql_lock_tables(), we
1442
* don't get the usual automatic call to StorageEngine::startStatement(), so
1443
* we manually call it here...
1445
to->s->getEngine()->startStatement(session);
1447
if (!(copy= new CopyField[to->getShare()->sizeFields()]))
1450
if (to->cursor->ha_external_lock(session, F_WRLCK))
1453
/* We need external lock before we can disable/enable keys */
1454
alter_table_manage_keys(session, to, from->cursor->indexes_are_disabled(), keys_onoff);
1456
/* We can abort alter table for any table type */
1457
session->abort_on_warning= !ignore;
1459
from->cursor->info(HA_STATUS_VARIABLE | HA_STATUS_NO_LOCK);
1460
to->cursor->ha_start_bulk_insert(from->cursor->stats.records);
1462
List_iterator<CreateField> it(create);
1465
for (Field **ptr= to->getFields(); *ptr ; ptr++)
1470
if (*ptr == to->next_number_field)
1471
auto_increment_field_copied= true;
1473
(copy_end++)->set(*ptr,def->field,0);
1478
found_count=delete_count=0;
1482
if (to->getShare()->hasPrimaryKey() && to->cursor->primary_key_is_clustered())
1484
char warn_buff[DRIZZLE_ERRMSG_SIZE];
1485
snprintf(warn_buff, sizeof(warn_buff),
1486
_("order_st BY ignored because there is a user-defined clustered "
1487
"index in the table '%-.192s'"),
1488
from->getMutableShare()->getTableName());
1489
push_warning(session, DRIZZLE_ERROR::WARN_LEVEL_WARN, ER_UNKNOWN_ERROR,
1494
from->sort.io_cache= new internal::IO_CACHE;
1496
memset(&tables, 0, sizeof(tables));
1498
tables.alias= tables.table_name= const_cast<char *>(from->getMutableShare()->getTableName());
1499
tables.db= const_cast<char *>(from->getMutableShare()->getSchemaName());
1502
if (session->lex->select_lex.setup_ref_array(session, order_num) ||
1503
setup_order(session, session->lex->select_lex.ref_pointer_array,
1504
&tables, fields, all_fields, order) ||
1505
!(sortorder= make_unireg_sortorder(order, &length, NULL)) ||
1506
(from->sort.found_records= filesort(session, from, sortorder, length,
1507
(optimizer::SqlSelect *) 0, HA_POS_ERROR,
1508
1, &examined_rows)) ==
1516
/* Tell handler that we have values for all columns in the to table */
1517
to->use_all_columns();
1518
info.init_read_record(session, from, (optimizer::SqlSelect *) 0, 1, true);
1520
to->cursor->extra(HA_EXTRA_IGNORE_DUP_KEY);
1521
session->row_count= 0;
1522
to->restoreRecordAsDefault(); // Create empty record
1523
while (!(error=info.read_record(&info)))
1525
if (session->killed)
1527
session->send_kill_message();
1531
session->row_count++;
1532
/* Return error if source table isn't empty. */
1533
if (error_if_not_empty)
1538
if (to->next_number_field)
1540
if (auto_increment_field_copied)
1541
to->auto_increment_field_not_null= true;
1543
to->next_number_field->reset();
1546
for (CopyField *copy_ptr=copy ; copy_ptr != copy_end ; copy_ptr++)
1548
copy_ptr->do_copy(copy_ptr);
1550
prev_insert_id= to->cursor->next_insert_id;
1551
error= to->cursor->insertRecord(to->record[0]);
1552
to->auto_increment_field_not_null= false;
1557
to->cursor->is_fatal_error(error, HA_CHECK_DUP))
1559
to->print_error(error, MYF(0));
1562
to->cursor->restore_auto_increment(prev_insert_id);
1571
info.end_read_record();
1572
from->free_io_cache();
1573
delete [] copy; // This is never 0
1575
if (to->cursor->ha_end_bulk_insert() && error <= 0)
1577
to->print_error(errno, MYF(0));
1580
to->cursor->extra(HA_EXTRA_NO_IGNORE_DUP_KEY);
1583
Ensure that the new table is saved properly to disk so that we
1586
if (transaction_services.autocommitOrRollback(session, false))
1588
if (! session->endActiveTransaction())
1592
session->abort_on_warning= 0;
1593
from->free_io_cache();
1594
*copied= found_count;
1595
*deleted=delete_count;
1596
to->cursor->ha_release_auto_increment();
1597
if (to->cursor->ha_external_lock(session,F_UNLCK))
1600
return(error > 0 ? -1 : 0);
1604
create_temporary_table(Session *session,
1605
TableIdentifier &identifier,
1606
HA_CREATE_INFO *create_info,
1607
message::Table &create_proto,
1608
AlterInfo *alter_info)
1613
Create a table with a temporary name.
1614
We don't log the statement, it will be logged later.
1616
create_proto.set_name(identifier.getTableName());
1618
create_proto.mutable_engine()->set_name(create_info->db_type->getName());
1620
error= mysql_create_table(session,
1622
create_info, create_proto, alter_info, true, 0, false);
1627
static Table *open_alter_table(Session *session, Table *table, TableIdentifier &identifier)
1631
/* Open the table so we need to copy the data to it. */
1632
if (table->getShare()->getType())
1635
tbl.db= const_cast<char *>(identifier.getSchemaName().c_str());
1636
tbl.alias= const_cast<char *>(identifier.getTableName().c_str());
1637
tbl.table_name= const_cast<char *>(identifier.getTableName().c_str());
1639
/* Table is in session->temporary_tables */
1640
new_table= session->openTable(&tbl, (bool*) 0, DRIZZLE_LOCK_IGNORE_FLUSH);
1644
/* Open our intermediate table */
1645
new_table= session->open_temporary_table(identifier, false);
1651
} /* namespace drizzled */