86
72
let's fetch the database default character set and
87
73
apply it to the table.
89
identifier::Schema identifier(db);
90
75
if (create_info->default_table_charset == NULL)
91
create_info->default_table_charset= plugin::StorageEngine::getSchemaCollation(identifier);
76
create_info->default_table_charset= get_default_db_collation(db);
80
Translate a cursor name to a table name (WL #1324).
83
filename_to_tablename()
86
to_length The size of the table name buffer.
91
uint32_t filename_to_tablename(const char *from, char *to, uint32_t to_length)
95
if (!memcmp(from, TMP_FILE_PREFIX, TMP_FILE_PREFIX_LENGTH))
97
/* Temporary table name. */
98
length= strlen(strncpy(to, from, to_length));
102
for (; *from && length < to_length; length++, from++)
109
/* We've found an escaped char - skip the @ */
112
/* There will be a two-position hex-char version of the char */
113
for (int x=1; x >= 0; x--)
115
if (*from >= '0' && *from <= '9')
116
to[length] += ((*from++ - '0') << (4 * x));
117
else if (*from >= 'a' && *from <= 'f')
118
to[length] += ((*from++ - 'a' + 10) << (4 * x));
120
/* Backup because we advanced extra in the inner loop */
130
Translate a table name to a cursor name (WL #1324).
133
tablename_to_filename()
135
to OUT The cursor name
136
to_length The size of the cursor name buffer.
139
true if errors happen. false on success.
141
bool tablename_to_filename(const char *from, char *to, size_t to_length)
145
for (; *from && length < to_length; length++, from++)
147
if ((*from >= '0' && *from <= '9') ||
148
(*from >= 'A' && *from <= 'Z') ||
149
(*from >= 'a' && *from <= 'z') ||
150
/* OSX defines an extra set of high-bit and multi-byte characters
151
that cannot be used on the filesystem. Instead of trying to sort
152
those out, we'll just escape encode all high-bit-set chars on OSX.
153
It won't really hurt anything - it'll just make some filenames ugly. */
154
#if !defined(TARGET_OS_OSX)
155
((unsigned char)*from >= 128) ||
165
if (length + 3 >= to_length)
168
/* We need to escape this char in a way that can be reversed */
170
to[length++]= hexchars[(*from >> 4) & 15];
171
to[length]= hexchars[(*from) & 15];
174
if (check_if_legal_tablename(to) &&
175
length + 4 < to_length)
177
memcpy(to + length, "@@@", 4);
185
Creates path to a cursor: drizzle_data_dir/db/table.ext
188
build_table_filename()
189
buff Where to write result
190
This may be the same as table_name.
193
table_name Table name
195
flags FN_FROM_IS_TMP or FN_TO_IS_TMP
196
table_name is temporary, do not change.
200
Uses database and table name, and extension to create
201
a cursor name in drizzle_data_dir. Database and table
202
names are converted from system_charset_info into "fscs".
203
Unless flags indicate a temporary table name.
204
'db' is always converted.
205
'ext' is not converted.
207
The conversion suppression is required for ALTER Table. This
208
statement creates intermediate tables. These are regular
209
(non-temporary) tables with a temporary name. Their path names must
210
be derivable from the table name. So we cannot use
211
build_tmptable_filename() for them.
214
path length on success, 0 on failure
217
size_t build_table_filename(char *buff, size_t bufflen, const char *db, const char *table_name, bool is_tmp)
219
char dbbuff[FN_REFLEN];
220
char tbbuff[FN_REFLEN];
221
bool conversion_error= false;
223
memset(tbbuff, 0, sizeof(tbbuff));
224
if (is_tmp) // FN_FROM_IS_TMP | FN_TO_IS_TMP
225
strncpy(tbbuff, table_name, sizeof(tbbuff));
228
conversion_error= tablename_to_filename(table_name, tbbuff, sizeof(tbbuff));
229
if (conversion_error)
231
errmsg_printf(ERRMSG_LVL_ERROR,
232
_("Table name cannot be encoded and fit within filesystem "
233
"name length restrictions."));
237
memset(dbbuff, 0, sizeof(dbbuff));
238
conversion_error= tablename_to_filename(db, dbbuff, sizeof(dbbuff));
239
if (conversion_error)
241
errmsg_printf(ERRMSG_LVL_ERROR,
242
_("Schema name cannot be encoded and fit within filesystem "
243
"name length restrictions."));
248
int rootdir_len= strlen(FN_ROOTDIR);
249
string table_path(drizzle_data_home);
250
int without_rootdir= table_path.length()-rootdir_len;
252
/* Don't add FN_ROOTDIR if dirzzle_data_home already includes it */
253
if (without_rootdir >= 0)
255
const char *tmp= table_path.c_str()+without_rootdir;
256
if (memcmp(tmp, FN_ROOTDIR, rootdir_len) != 0)
257
table_path.append(FN_ROOTDIR);
260
table_path.append(dbbuff);
261
table_path.append(FN_ROOTDIR);
262
table_path.append(tbbuff);
264
if (bufflen < table_path.length())
267
strcpy(buff, table_path.c_str());
269
return table_path.length();
274
Creates path to a cursor: drizzle_tmpdir/#sql1234_12_1.ext
277
build_tmptable_filename()
278
session The thread handle.
279
buff Where to write result
284
Uses current_pid, thread_id, and tmp_table counter to create
285
a cursor name in drizzle_tmpdir.
288
path length on success, 0 on failure
291
size_t build_tmptable_filename(char *buff, size_t bufflen)
294
ostringstream path_str, post_tmpdir_str;
297
Session *session= current_session;
299
path_str << drizzle_tmpdir;
300
post_tmpdir_str << "/" << TMP_FILE_PREFIX << current_pid;
301
post_tmpdir_str << session->thread_id << session->tmp_table++;
302
tmp= post_tmpdir_str.str();
304
transform(tmp.begin(), tmp.end(), tmp.begin(), ::tolower);
308
if (bufflen < path_str.str().length())
311
length= unpack_filename(buff, path_str.str().c_str());
139
386
-1 Thread was killed
142
int rm_table_part2(Session *session, TableList *tables, bool if_exists,
389
int mysql_rm_table_part2(Session *session, TableList *tables, bool if_exists,
145
392
TableList *table;
393
char path[FN_REFLEN];
394
uint32_t path_length= 0;
146
395
String wrong_tables;
148
397
bool foreign_key_error= false;
152
boost::mutex::scoped_lock scopedLock(table::Cache::singleton().mutex());
154
if (not drop_temporary && session->lock_table_names_exclusively(tables))
159
/* Don't give warnings for not found errors, as we already generate notes */
160
session->no_warnings_for_error= 1;
162
for (table= tables; table; table= table->next_local)
164
identifier::Table tmp_identifier(table->getSchemaName(), table->getTableName());
166
error= session->drop_temporary_table(tmp_identifier);
170
// removed temporary table
399
pthread_mutex_lock(&LOCK_open); /* Part 2 of rm a table */
402
If we have the table in the definition cache, we don't have to check the
403
.frm cursor to find if the table is a normal table (not view) and what
407
for (table= tables; table; table= table->next_local)
410
table->db_type= NULL;
411
if ((share= TableShare::getShare(table->db, table->table_name)))
412
table->db_type= share->db_type();
415
if (!drop_temporary && lock_table_names_exclusively(session, tables))
417
pthread_mutex_unlock(&LOCK_open);
421
/* Don't give warnings for not found errors, as we already generate notes */
422
session->no_warnings_for_error= 1;
424
for (table= tables; table; table= table->next_local)
427
plugin::StorageEngine *table_type;
429
error= session->drop_temporary_table(table);
433
// removed temporary table
437
goto err_with_placeholders;
439
// temporary table not found
443
table_type= table->db_type;
444
if (drop_temporary == false)
447
abort_locked_tables(session, db, table->table_name);
448
remove_table_from_cache(session, db, table->table_name,
449
RTFC_WAIT_OTHER_THREAD_FLAG |
450
RTFC_CHECK_KILLED_FLAG);
452
If the table was used in lock tables, remember it so that
453
unlock_table_names can free it
455
if ((locked_table= drop_locked_tables(session, db, table->table_name)))
456
table->table= locked_table;
461
goto err_with_placeholders;
463
/* remove .frm cursor and engine files */
464
path_length= build_table_filename(path, sizeof(path), db, table->table_name, table->internal_tmp_table);
466
TableIdentifier identifier(db, table->table_name, table->internal_tmp_table ? INTERNAL_TMP_TABLE : NO_TMP_TABLE);
468
if (drop_temporary ||
470
&& (plugin::StorageEngine::getTableDefinition(*session,
471
identifier) != EEXIST))))
473
// Table was not found on disk and table can't be created from engine
475
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
476
ER_BAD_TABLE_ERROR, ER(ER_BAD_TABLE_ERROR),
176
// temporary table not found
180
if (drop_temporary == false)
183
abort_locked_tables(session, tmp_identifier);
184
table::Cache::singleton().removeTable(session, tmp_identifier,
185
RTFC_WAIT_OTHER_THREAD_FLAG |
186
RTFC_CHECK_KILLED_FLAG);
188
If the table was used in lock tables, remember it so that
189
unlock_table_names can free it
191
if ((locked_table= drop_locked_tables(session, tmp_identifier)))
192
table->table= locked_table;
194
if (session->getKilled())
200
identifier::Table identifier(table->getSchemaName(), table->getTableName(), table->getInternalTmpTable() ? message::Table::INTERNAL : message::Table::STANDARD);
202
if (drop_temporary || not plugin::StorageEngine::doesTableExist(*session, identifier))
204
// Table was not found on disk and table can't be created from engine
206
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
207
ER_BAD_TABLE_ERROR, ER(ER_BAD_TABLE_ERROR),
208
table->getTableName());
216
drizzled::error_t local_error;
218
/* Generate transaction event ONLY when we successfully drop */
219
if (plugin::StorageEngine::dropTable(*session, identifier, local_error))
221
TransactionServices &transaction_services= TransactionServices::singleton();
222
transaction_services.dropTable(*session, identifier, if_exists);
226
if (local_error == HA_ERR_NO_SUCH_TABLE and if_exists)
229
session->clear_error();
232
if (local_error == HA_ERR_ROW_IS_REFERENCED)
234
/* the table is referenced by a foreign key constraint */
235
foreign_key_error= true;
243
if (wrong_tables.length())
244
wrong_tables.append(',');
245
wrong_tables.append(String(table->getTableName(), system_charset_info));
249
tables->unlock_table_names();
483
error= plugin::StorageEngine::dropTable(*session,
487
if ((error == ENOENT || error == HA_ERR_NO_SUCH_TABLE) && if_exists)
490
session->clear_error();
493
if (error == HA_ERR_ROW_IS_REFERENCED)
495
/* the table is referenced by a foreign key constraint */
496
foreign_key_error= true;
500
if (error == 0 || (if_exists && foreign_key_error == false))
501
write_bin_log_drop_table(session, if_exists, db, table->table_name);
505
if (wrong_tables.length())
506
wrong_tables.append(',');
507
wrong_tables.append(String(table->table_name,system_charset_info));
511
It's safe to unlock LOCK_open: we have an exclusive lock
514
pthread_mutex_unlock(&LOCK_open);
253
516
if (wrong_tables.length())
255
if (not foreign_key_error)
518
if (!foreign_key_error)
257
519
my_printf_error(ER_BAD_TABLE_ERROR, ER(ER_BAD_TABLE_ERROR), MYF(0),
258
520
wrong_tables.c_ptr());
262
523
my_message(ER_ROW_IS_REFERENCED, ER(ER_ROW_IS_REFERENCED), MYF(0));
1429
bool create_table_no_lock(Session *session,
1430
const identifier::Table &identifier,
1541
bool mysql_create_table_no_lock(Session *session,
1542
TableIdentifier &identifier,
1431
1543
HA_CREATE_INFO *create_info,
1432
message::Table &table_proto,
1544
message::Table *table_proto,
1433
1545
AlterInfo *alter_info,
1434
1546
bool internal_tmp_table,
1435
1547
uint32_t select_field_count,
1436
1548
bool is_if_not_exists)
1438
1550
uint db_options, key_count;
1439
KeyInfo *key_info_buffer;
1551
KEY *key_info_buffer;
1440
1553
bool error= true;
1555
bool lex_identified_temp_table=
1556
(table_proto->type() == drizzled::message::Table::TEMPORARY);
1442
1558
/* Check for duplicate fields and check type of table to create */
1443
if (not alter_info->create_list.elements)
1559
if (!alter_info->create_list.elements)
1445
1561
my_message(ER_TABLE_MUST_HAVE_COLUMNS, ER(ER_TABLE_MUST_HAVE_COLUMNS),
1449
assert(identifier.getTableName() == table_proto.name());
1565
assert(strcmp(identifier.getTableName(), table_proto->name().c_str())==0);
1450
1566
db_options= create_info->table_options;
1452
set_table_default_charset(create_info, identifier.getSchemaName().c_str());
1454
/* Build a Table object to pass down to the engine, and the do the actual create. */
1455
if (not prepare_create_table(session, create_info, table_proto, alter_info,
1458
&key_info_buffer, &key_count,
1459
select_field_count))
1461
boost_unique_lock_t lock(table::Cache::singleton().mutex()); /* CREATE TABLE (some confussion on naming, double check) */
1462
error= locked_create_event(session,
1469
db_options, key_count,
1473
session->set_proc_info("After create");
1479
@note the following two methods implement create [temporary] table.
1481
static bool drizzle_create_table(Session *session,
1482
const identifier::Table &identifier,
1483
HA_CREATE_INFO *create_info,
1484
message::Table &table_proto,
1485
AlterInfo *alter_info,
1486
bool internal_tmp_table,
1487
uint32_t select_field_count,
1488
bool is_if_not_exists)
1490
Table *name_lock= NULL;
1493
if (session->lock_table_name_if_not_cached(identifier, &name_lock))
1497
else if (name_lock == NULL)
1567
if (create_info->row_type == ROW_TYPE_DYNAMIC)
1568
db_options|=HA_OPTION_PACK_RECORD;
1569
if (!(cursor= create_info->db_type->getCursor(share, session->mem_root)))
1571
my_error(ER_OUTOFMEMORY, MYF(0), sizeof(Cursor));
1575
set_table_default_charset(create_info, identifier.getDBName());
1577
/* Check if table exists */
1578
if (mysql_prepare_create_table(session, create_info, table_proto, alter_info,
1580
&db_options, cursor,
1581
&key_info_buffer, &key_count,
1582
select_field_count))
1585
/* Check if table already exists */
1586
if (lex_identified_temp_table &&
1587
session->find_temporary_table(identifier.getDBName(), identifier.getTableName()))
1499
1589
if (is_if_not_exists)
1591
create_info->table_existed= 1; // Mark that table existed
1501
1592
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1502
1593
ER_TABLE_EXISTS_ERROR, ER(ER_TABLE_EXISTS_ERROR),
1503
identifier.getTableName().c_str());
1504
create_info->table_existed= 1;
1509
my_error(ER_TABLE_EXISTS_ERROR, identifier);
1515
result= create_table_no_lock(session,
1527
boost_unique_lock_t lock(table::Cache::singleton().mutex()); /* Lock for removing name_lock during table create */
1528
session->unlink_open_table(name_lock);
1594
identifier.getTableName());
1598
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), identifier.getTableName());
1602
pthread_mutex_lock(&LOCK_open); /* CREATE TABLE (some confussion on naming, double check) */
1603
if (!internal_tmp_table && ! lex_identified_temp_table)
1605
if (plugin::StorageEngine::getTableDefinition(*session,
1606
identifier)==EEXIST)
1608
if (is_if_not_exists)
1611
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1612
ER_TABLE_EXISTS_ERROR, ER(ER_TABLE_EXISTS_ERROR),
1613
identifier.getTableName());
1614
create_info->table_existed= 1; // Mark that table existed
1617
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), identifier.getTableName());
1619
goto unlock_and_end;
1622
We don't assert here, but check the result, because the table could be
1623
in the table definition cache and in the same time the .frm could be
1624
missing from the disk, in case of manual intervention which deletes
1625
the .frm cursor. The user has to use FLUSH TABLES; to clear the cache.
1626
Then she could create the table. This case is pretty obscure and
1627
therefore we don't introduce a new error message only for it.
1629
if (TableShare::getShare(identifier.getDBName(), identifier.getTableName()))
1631
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), identifier.getTableName());
1632
goto unlock_and_end;
1637
Check that table with given name does not already
1638
exist in any storage engine. In such a case it should
1639
be discovered and the error ER_TABLE_EXISTS_ERROR be returned
1640
unless user specified CREATE TABLE IF EXISTS
1641
The LOCK_open mutex has been locked to make sure no
1642
one else is attempting to discover the table. Since
1643
it's not on disk as a frm cursor, no one could be using it!
1645
if (! lex_identified_temp_table)
1647
int retcode= plugin::StorageEngine::getTableDefinition(*session, identifier);
1652
/* Normal case, no table exists. we can go and create it */
1655
if (is_if_not_exists)
1658
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1659
ER_TABLE_EXISTS_ERROR, ER(ER_TABLE_EXISTS_ERROR),
1660
identifier.getTableName());
1661
create_info->table_existed= 1; // Mark that table existed
1662
goto unlock_and_end;
1664
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), identifier.getTableName());
1665
goto unlock_and_end;
1667
my_error(retcode, MYF(0), identifier.getTableName());
1668
goto unlock_and_end;
1672
session->set_proc_info("creating table");
1673
create_info->table_existed= 0; // Mark that table is created
1675
create_info->table_options= db_options;
1677
if (rea_create_table(session, identifier,
1679
create_info, alter_info->create_list,
1680
key_count, key_info_buffer))
1681
goto unlock_and_end;
1683
if (lex_identified_temp_table)
1685
/* Open table and put in temporary table list */
1686
if (!(session->open_temporary_table(identifier)))
1688
(void) session->rm_temporary_table(create_info->db_type, identifier);
1689
goto unlock_and_end;
1694
Don't write statement if:
1695
- It is an internal temporary table,
1696
- Row-based logging is used and it we are creating a temporary table, or
1697
- The binary log is not open.
1698
Otherwise, the statement shall be binlogged.
1700
if (!internal_tmp_table && ! lex_identified_temp_table)
1701
write_bin_log(session, session->query, session->query_length);
1704
pthread_mutex_unlock(&LOCK_open);
1707
session->set_proc_info("After create");
1536
Database locking aware wrapper for create_table_no_lock(),
1714
Database locking aware wrapper for mysql_create_table_no_lock(),
1538
bool create_table(Session *session,
1539
const identifier::Table &identifier,
1717
bool mysql_create_table(Session *session,
1718
TableIdentifier &identifier,
1540
1719
HA_CREATE_INFO *create_info,
1541
message::Table &table_proto,
1720
message::Table *table_proto,
1542
1721
AlterInfo *alter_info,
1543
1722
bool internal_tmp_table,
1544
1723
uint32_t select_field_count,
1545
1724
bool is_if_not_exists)
1547
if (identifier.isTmp())
1549
return create_table_no_lock(session,
1559
return drizzle_create_table(session,
1726
Table *name_lock= NULL;
1728
bool lex_identified_temp_table=
1729
(table_proto->type() == drizzled::message::Table::TEMPORARY);
1731
if (! lex_identified_temp_table)
1733
if (session->lock_table_name_if_not_cached(identifier.getDBName(),
1734
identifier.getTableName(),
1740
if (name_lock == NULL)
1742
if (is_if_not_exists)
1744
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1745
ER_TABLE_EXISTS_ERROR, ER(ER_TABLE_EXISTS_ERROR),
1746
identifier.getTableName());
1747
create_info->table_existed= 1;
1752
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), identifier.getTableName());
1759
result= mysql_create_table_no_lock(session,
1771
pthread_mutex_lock(&LOCK_open); /* Lock for removing name_lock during table create */
1772
session->unlink_open_table(name_lock);
1773
pthread_mutex_unlock(&LOCK_open);
2189
Create a table identical to the specified table
2192
mysql_create_like_table()
2193
session Thread object
2194
table Table list element for target table
2195
src_table Table list element for source table
2196
create_info Create info
2203
bool mysql_create_like_table(Session* session, TableList* table, TableList* src_table,
2204
drizzled::message::Table& create_table_proto,
2205
drizzled::plugin::StorageEngine *engine_arg,
2206
bool is_if_not_exists,
2209
Table *name_lock= 0;
2210
char src_path[FN_REFLEN];
2211
char *db= table->db;
2212
char *table_name= table->table_name;
2216
message::Table src_proto;
2217
bool lex_identified_temp_table=
2218
(create_table_proto.type() == drizzled::message::Table::TEMPORARY);
2221
By opening source table we guarantee that it exists and no concurrent
2222
DDL operation will mess with it. Later we also take an exclusive
2223
name-lock on target table name, which makes copying of .frm cursor,
2224
call to plugin::StorageEngine::createTable() and binlogging atomic
2225
against concurrent DML and DDL operations on target table.
2226
Thus by holding both these "locks" we ensure that our statement is
2227
properly isolated from all concurrent operations which matter.
2229
if (session->open_tables_from_list(&src_table, ¬_used))
2232
strncpy(src_path, src_table->table->s->path.str, sizeof(src_path));
2235
TableIdentifier destination_identifier(db, table_name, lex_identified_temp_table ? TEMP_TABLE : NO_TMP_TABLE);
2238
Check that destination tables does not exist. Note that its name
2239
was already checked when it was added to the table list.
2241
if (lex_identified_temp_table)
2243
if (session->find_temporary_table(db, table_name))
2248
if (session->lock_table_name_if_not_cached(db, table_name, &name_lock))
2253
if (plugin::StorageEngine::getTableDefinition(*session,
2254
destination_identifier) == EEXIST)
1981
2259
Create a new table by copying from source table
1983
2261
Altough exclusive name-lock on target table protects us from concurrent
1984
2262
DML and DDL operations on it we still want to wrap .FRM creation and call
1985
2263
to plugin::StorageEngine::createTable() in critical section protected by
1986
table::Cache::singleton().mutex() in order to provide minimal atomicity against operations which
2264
LOCK_open in order to provide minimal atomicity against operations which
1987
2265
disregard name-locks, like I_S implementation, for example. This is a
1988
2266
temporary and should not be copied. Instead we should fix our code to
1989
2267
always honor name-locks.
1991
Also some engines (e.g. NDB cluster) require that table::Cache::singleton().mutex() should be held
2269
Also some engines (e.g. NDB cluster) require that LOCK_open should be held
1992
2270
during the call to plugin::StorageEngine::createTable().
1993
2271
See bug #28614 for more info.
1995
static bool create_table_wrapper(Session &session,
1996
const message::Table& create_table_proto,
1997
identifier::Table::const_reference destination_identifier,
1998
identifier::Table::const_reference source_identifier,
2001
// We require an additional table message because during parsing we used
2002
// a "new" message and it will not have all of the information that the
2003
// source table message would have.
2004
message::Table new_table_message;
2005
drizzled::error_t error;
2007
message::table::shared_ptr source_table_message= plugin::StorageEngine::getTableMessage(session, source_identifier, error);
2009
if (not source_table_message)
2011
my_error(ER_TABLE_UNKNOWN, source_identifier);
2015
new_table_message.CopyFrom(*source_table_message);
2017
if (destination_identifier.isTmp())
2019
new_table_message.set_type(message::Table::TEMPORARY);
2273
pthread_mutex_lock(&LOCK_open); /* We lock for CREATE TABLE LIKE to copy table definition */
2275
int protoerr= EEXIST;
2278
* If an engine was not specified and we are reading from an I_S table, then we need to toss an
2279
* error. This should go away soon.
2280
* @todo make this go away!
2282
if (! is_engine_set)
2284
string tab_name(src_path);
2285
string i_s_prefix("./information_schema/");
2286
if (tab_name.compare(0, i_s_prefix.length(), i_s_prefix) == 0)
2288
pthread_mutex_unlock(&LOCK_open);
2289
my_error(ER_ILLEGAL_HA_CREATE_OPTION,
2291
"INFORMATION_ENGINE",
2297
protoerr= plugin::StorageEngine::getTableDefinition(*session,
2304
message::Table new_proto(src_proto);
2306
if (lex_identified_temp_table)
2308
new_proto.set_type(message::Table::TEMPORARY);
2312
new_proto.set_type(message::Table::STANDARD);
2317
message::Table::StorageEngine *protoengine;
2319
protoengine= new_proto.mutable_engine();
2320
protoengine->set_name(create_table_proto.engine().name());
2323
if (protoerr == EEXIST)
2325
plugin::StorageEngine* engine= plugin::StorageEngine::findByName(*session,
2326
new_proto.engine().name());
2328
if (engine->check_flag(HTON_BIT_HAS_DATA_DICTIONARY) == false)
2330
string dst_proto_path(destination_identifier.getPath());
2331
dst_proto_path.append(".dfe");
2333
protoerr= drizzle_write_proto_file(dst_proto_path.c_str(), &new_proto);
2343
if (my_errno == ENOENT)
2344
my_error(ER_BAD_DB_ERROR,MYF(0),db);
2346
my_error(ER_CANT_CREATE_FILE, MYF(0), destination_identifier.getPath(), my_errno);
2347
pthread_mutex_unlock(&LOCK_open);
2352
As mysql_truncate don't work on a new table at this stage of
2353
creation, instead create the table directly (for both normal
2354
and temporary tables).
2356
err= plugin::StorageEngine::createTable(*session,
2357
destination_identifier,
2360
pthread_mutex_unlock(&LOCK_open);
2362
if (lex_identified_temp_table)
2364
if (err || !session->open_temporary_table(destination_identifier))
2366
(void) session->rm_temporary_table(engine_arg, destination_identifier);
2372
TableIdentifier identifier(db, table_name, NO_TMP_TABLE);
2373
quick_rm_table(*session, identifier);
2379
We have to write the query before we unlock the tables.
2383
Since temporary tables are not replicated under row-based
2384
replication, CREATE TABLE ... LIKE ... needs special
2385
treatement. We have four cases to consider, according to the
2386
following decision table:
2388
==== ========= ========= ==============================
2389
Case Target Source Write to binary log
2390
==== ========= ========= ==============================
2391
1 normal normal Original statement
2392
2 normal temporary Generated statement
2393
3 temporary normal Nothing
2394
4 temporary temporary Nothing
2395
==== ========= ========= ==============================
2397
if (! lex_identified_temp_table)
2399
if (src_table->table->s->tmp_table) // Case 2
2402
String query(buf, sizeof(buf), system_charset_info);
2403
query.length(0); // Have to zero it since constructor doesn't
2407
Here we open the destination table, on which we already have
2408
name-lock. This is needed for store_create_info() to work.
2409
The table will be closed by unlink_open_table() at the end
2412
table->table= name_lock;
2413
pthread_mutex_lock(&LOCK_open); /* Open new table we have just acquired */
2414
if (session->reopen_name_locked_table(table, false))
2416
pthread_mutex_unlock(&LOCK_open);
2419
pthread_mutex_unlock(&LOCK_open);
2421
int result= store_create_info(table, &query, is_if_not_exists);
2423
assert(result == 0); // store_create_info() always return 0
2424
write_bin_log(session, query.ptr(), query.length());
2427
write_bin_log(session, session->query, session->query_length);
2435
if (is_if_not_exists)
2437
char warn_buff[DRIZZLE_ERRMSG_SIZE];
2438
snprintf(warn_buff, sizeof(warn_buff),
2439
ER(ER_TABLE_EXISTS_ERROR), table_name);
2440
push_warning(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
2441
ER_TABLE_EXISTS_ERROR,warn_buff);
2023
new_table_message.set_type(message::Table::STANDARD);
2028
new_table_message.mutable_engine()->set_name(create_table_proto.engine().name());
2031
{ // We now do a selective copy of elements on to the new table.
2032
new_table_message.set_name(create_table_proto.name());
2033
new_table_message.set_schema(create_table_proto.schema());
2034
new_table_message.set_catalog(create_table_proto.catalog());
2037
/* Fix names of foreign keys being added */
2038
for (int32_t j= 0; j < new_table_message.fk_constraint_size(); j++)
2040
if (new_table_message.fk_constraint(j).has_name())
2042
std::string name(new_table_message.name());
2045
name.append("_ibfk_");
2046
snprintf(number, sizeof(number), "%d", j+1);
2047
name.append(number);
2049
message::Table::ForeignKeyConstraint *pfkey= new_table_message.mutable_fk_constraint(j);
2050
pfkey->set_name(name);
2055
As mysql_truncate don't work on a new table at this stage of
2056
creation, instead create the table directly (for both normal and temporary tables).
2058
bool success= plugin::StorageEngine::createTable(session,
2059
destination_identifier,
2062
if (success && not destination_identifier.isTmp())
2064
TransactionServices &transaction_services= TransactionServices::singleton();
2065
transaction_services.createTable(session, new_table_message);
2072
Create a table identical to the specified table
2076
session Thread object
2077
table Table list element for target table
2078
src_table Table list element for source table
2079
create_info Create info
2086
bool create_like_table(Session* session,
2087
identifier::Table::const_reference destination_identifier,
2088
identifier::Table::const_reference source_identifier,
2089
message::Table &create_table_proto,
2090
bool is_if_not_exists,
2094
bool table_exists= false;
2097
Check that destination tables does not exist. Note that its name
2098
was already checked when it was added to the table list.
2100
For temporary tables we don't aim to grab locks.
2102
if (destination_identifier.isTmp())
2104
if (session->find_temporary_table(destination_identifier))
2110
bool was_created= create_table_wrapper(*session,
2112
destination_identifier,
2115
if (not was_created) // This is pretty paranoid, but we assume something might not clean up after itself
2117
(void) session->rm_temporary_table(destination_identifier, true);
2119
else if (not session->open_temporary_table(destination_identifier))
2121
// We created, but we can't open... also, a hack.
2122
(void) session->rm_temporary_table(destination_identifier, true);
2130
else // Standard table which will require locks.
2132
Table *name_lock= 0;
2134
if (session->lock_table_name_if_not_cached(destination_identifier, &name_lock))
2138
boost_unique_lock_t lock(table::Cache::singleton().mutex()); /* unlink open tables for create table like*/
2139
session->unlink_open_table(name_lock);
2149
else if (plugin::StorageEngine::doesTableExist(*session, destination_identifier))
2153
else // Otherwise we create the table
2157
boost_unique_lock_t lock(table::Cache::singleton().mutex()); /* We lock for CREATE TABLE LIKE to copy table definition */
2158
was_created= create_table_wrapper(*session, create_table_proto, destination_identifier,
2159
source_identifier, is_engine_set);
2162
// So we blew the creation of the table, and we scramble to clean up
2163
// anything that might have been created (read... it is a hack)
2164
if (not was_created)
2166
plugin::StorageEngine::dropTable(*session, destination_identifier);
2176
boost_unique_lock_t lock(table::Cache::singleton().mutex()); /* unlink open tables for create table like*/
2177
session->unlink_open_table(name_lock);
2183
if (is_if_not_exists)
2185
char warn_buff[DRIZZLE_ERRMSG_SIZE];
2186
snprintf(warn_buff, sizeof(warn_buff),
2187
ER(ER_TABLE_EXISTS_ERROR), destination_identifier.getTableName().c_str());
2188
push_warning(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
2189
ER_TABLE_EXISTS_ERROR, warn_buff);
2193
my_error(ER_TABLE_EXISTS_ERROR, destination_identifier);
2202
bool analyze_table(Session* session, TableList* tables, HA_CHECK_OPT* check_opt)
2445
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), table_name);
2450
pthread_mutex_lock(&LOCK_open); /* unlink open tables for create table like*/
2451
session->unlink_open_table(name_lock);
2452
pthread_mutex_unlock(&LOCK_open);
2458
bool mysql_analyze_table(Session* session, TableList* tables, HA_CHECK_OPT* check_opt)
2204
2460
thr_lock_type lock_type = TL_READ_NO_INSERT;
2206
return(admin_table(session, tables, check_opt,
2462
return(mysql_admin_table(session, tables, check_opt,
2207
2463
"analyze", lock_type, true,
2208
2464
&Cursor::ha_analyze));
2212
bool check_table(Session* session, TableList* tables,HA_CHECK_OPT* check_opt)
2468
bool mysql_check_table(Session* session, TableList* tables,HA_CHECK_OPT* check_opt)
2214
2470
thr_lock_type lock_type = TL_READ_NO_INSERT;
2216
return(admin_table(session, tables, check_opt,
2472
return(mysql_admin_table(session, tables, check_opt,
2217
2473
"check", lock_type,
2219
2475
&Cursor::ha_check));
2222
} /* namespace drizzled */
2479
bool mysql_checksum_table(Session *session, TableList *tables,
2483
List<Item> field_list;
2486
field_list.push_back(item = new Item_empty_string("Table", NAME_LEN*2));
2487
item->maybe_null= 1;
2488
field_list.push_back(item= new Item_int("Checksum", (int64_t) 1,
2489
MY_INT64_NUM_DECIMAL_DIGITS));
2490
item->maybe_null= 1;
2491
if (session->client->sendFields(&field_list))
2494
/* Open one table after the other to keep lock time as short as possible. */
2495
for (table= tables; table; table= table->next_local)
2497
char table_name[NAME_LEN*2+2];
2500
sprintf(table_name,"%s.%s",table->db,table->table_name);
2502
t= table->table= session->openTableLock(table, TL_READ);
2503
session->clear_error(); // these errors shouldn't get client
2505
session->client->store(table_name);
2509
/* Table didn't exist */
2510
session->client->store();
2511
session->clear_error();
2516
@note if the engine keeps a checksum then we return the checksum, otherwise we calculate
2518
if (t->cursor->getEngine()->check_flag(HTON_BIT_HAS_CHECKSUM))
2520
session->client->store((uint64_t)t->cursor->checksum());
2524
/* calculating table's checksum */
2526
unsigned char null_mask=256 - (1 << t->s->last_null_bit_pos);
2528
t->use_all_columns();
2530
if (t->cursor->ha_rnd_init(1))
2531
session->client->store();
2536
ha_checksum row_crc= 0;
2537
int error= t->cursor->rnd_next(t->record[0]);
2538
if (unlikely(error))
2540
if (error == HA_ERR_RECORD_DELETED)
2544
if (t->s->null_bytes)
2546
/* fix undefined null bits */
2547
t->record[0][t->s->null_bytes-1] |= null_mask;
2548
if (!(t->s->db_create_options & HA_OPTION_PACK_RECORD))
2549
t->record[0][0] |= 1;
2551
row_crc= my_checksum(row_crc, t->record[0], t->s->null_bytes);
2554
for (uint32_t i= 0; i < t->s->fields; i++ )
2556
Field *f= t->field[i];
2557
if ((f->type() == DRIZZLE_TYPE_BLOB) ||
2558
(f->type() == DRIZZLE_TYPE_VARCHAR))
2562
row_crc= my_checksum(row_crc, (unsigned char*) tmp.ptr(), tmp.length());
2565
row_crc= my_checksum(row_crc, f->ptr,
2571
session->client->store((uint64_t)crc);
2572
t->cursor->ha_rnd_end();
2575
session->clear_error();
2576
session->close_thread_tables();
2577
table->table=0; // For query cache
2579
if (session->client->flush())
2587
session->close_thread_tables(); // Shouldn't be needed