123
195
sql_command_flags[SQLCOM_DROP_INDEX]= CF_CHANGES_DATA;
125
197
sql_command_flags[SQLCOM_UPDATE]= CF_CHANGES_DATA | CF_HAS_ROW_COUNT;
198
sql_command_flags[SQLCOM_UPDATE_MULTI]= CF_CHANGES_DATA | CF_HAS_ROW_COUNT;
126
199
sql_command_flags[SQLCOM_INSERT]= CF_CHANGES_DATA | CF_HAS_ROW_COUNT;
127
200
sql_command_flags[SQLCOM_INSERT_SELECT]= CF_CHANGES_DATA | CF_HAS_ROW_COUNT;
128
201
sql_command_flags[SQLCOM_DELETE]= CF_CHANGES_DATA | CF_HAS_ROW_COUNT;
202
sql_command_flags[SQLCOM_DELETE_MULTI]= CF_CHANGES_DATA | CF_HAS_ROW_COUNT;
129
203
sql_command_flags[SQLCOM_REPLACE]= CF_CHANGES_DATA | CF_HAS_ROW_COUNT;
130
204
sql_command_flags[SQLCOM_REPLACE_SELECT]= CF_CHANGES_DATA | CF_HAS_ROW_COUNT;
206
sql_command_flags[SQLCOM_SHOW_STATUS]= CF_STATUS_COMMAND;
207
sql_command_flags[SQLCOM_SHOW_DATABASES]= CF_STATUS_COMMAND;
208
sql_command_flags[SQLCOM_SHOW_OPEN_TABLES]= CF_STATUS_COMMAND;
132
209
sql_command_flags[SQLCOM_SHOW_FIELDS]= CF_STATUS_COMMAND;
133
210
sql_command_flags[SQLCOM_SHOW_KEYS]= CF_STATUS_COMMAND;
211
sql_command_flags[SQLCOM_SHOW_VARIABLES]= CF_STATUS_COMMAND;
212
sql_command_flags[SQLCOM_SHOW_CHARSETS]= CF_STATUS_COMMAND;
213
sql_command_flags[SQLCOM_SHOW_COLLATIONS]= CF_STATUS_COMMAND;
214
sql_command_flags[SQLCOM_SHOW_NEW_MASTER]= CF_STATUS_COMMAND;
215
sql_command_flags[SQLCOM_SHOW_BINLOGS]= CF_STATUS_COMMAND;
216
sql_command_flags[SQLCOM_SHOW_SLAVE_HOSTS]= CF_STATUS_COMMAND;
217
sql_command_flags[SQLCOM_SHOW_BINLOG_EVENTS]= CF_STATUS_COMMAND;
134
218
sql_command_flags[SQLCOM_SHOW_WARNS]= CF_STATUS_COMMAND;
135
219
sql_command_flags[SQLCOM_SHOW_ERRORS]= CF_STATUS_COMMAND;
220
sql_command_flags[SQLCOM_SHOW_ENGINE_STATUS]= CF_STATUS_COMMAND;
221
sql_command_flags[SQLCOM_SHOW_ENGINE_MUTEX]= CF_STATUS_COMMAND;
222
sql_command_flags[SQLCOM_SHOW_ENGINE_LOGS]= CF_STATUS_COMMAND;
223
sql_command_flags[SQLCOM_SHOW_PROCESSLIST]= CF_STATUS_COMMAND;
136
224
sql_command_flags[SQLCOM_SHOW_CREATE_DB]= CF_STATUS_COMMAND;
137
225
sql_command_flags[SQLCOM_SHOW_CREATE]= CF_STATUS_COMMAND;
226
sql_command_flags[SQLCOM_SHOW_MASTER_STAT]= CF_STATUS_COMMAND;
227
sql_command_flags[SQLCOM_SHOW_SLAVE_STAT]= CF_STATUS_COMMAND;
229
sql_command_flags[SQLCOM_SHOW_TABLES]= (CF_STATUS_COMMAND |
230
CF_SHOW_TABLE_COMMAND);
231
sql_command_flags[SQLCOM_SHOW_TABLE_STATUS]= (CF_STATUS_COMMAND |
232
CF_SHOW_TABLE_COMMAND);
235
The following is used to preserver CF_ROW_COUNT during the
236
a CALL or EXECUTE statement, so the value generated by the
237
last called (or executed) statement is preserved.
238
See mysql_execute_command() for how CF_ROW_COUNT is used.
240
sql_command_flags[SQLCOM_EXECUTE]= CF_HAS_ROW_COUNT;
140
243
The following admin table operations are allowed
246
sql_command_flags[SQLCOM_REPAIR]= CF_WRITE_LOGS_COMMAND;
247
sql_command_flags[SQLCOM_OPTIMIZE]= CF_WRITE_LOGS_COMMAND;
143
248
sql_command_flags[SQLCOM_ANALYZE]= CF_WRITE_LOGS_COMMAND;
252
bool is_update_query(enum enum_sql_command command)
254
DBUG_ASSERT(command >= 0 && command <= SQLCOM_END);
255
return (sql_command_flags[command] & CF_CHANGES_DATA) != 0;
258
void execute_init_command(THD *thd, sys_var_str *init_command_var,
259
rw_lock_t *var_mutex)
262
ulong save_client_capabilities;
264
thd_proc_info(thd, "Execution of init_command");
266
We need to lock init_command_var because
267
during execution of init_command_var query
268
values of init_command_var can't be changed
270
rw_rdlock(var_mutex);
271
save_client_capabilities= thd->client_capabilities;
272
thd->client_capabilities|= CLIENT_MULTI_QUERIES;
274
We don't need return result of execution to client side.
275
To forbid this we should set thd->net.vio to 0.
277
save_vio= thd->net.vio;
279
dispatch_command(COM_QUERY, thd,
280
init_command_var->value,
281
init_command_var->value_length);
282
rw_unlock(var_mutex);
283
thd->client_capabilities= save_client_capabilities;
284
thd->net.vio= save_vio;
289
Execute commands from bootstrap_file.
291
Used when creating the initial grant tables.
294
pthread_handler_t handle_bootstrap(void *arg)
297
FILE *file=bootstrap_file;
299
const char* found_semicolon= NULL;
301
/* The following must be called before DBUG_ENTER */
302
thd->thread_stack= (char*) &thd;
303
if (my_thread_init() || thd->store_globals())
305
close_connection(thd, ER_OUT_OF_RESOURCES, 1);
309
DBUG_ENTER("handle_bootstrap");
311
pthread_detach_this_thread();
312
thd->thread_stack= (char*) &thd;
314
if (thd->variables.max_join_size == HA_POS_ERROR)
315
thd->options |= OPTION_BIG_SELECTS;
317
thd_proc_info(thd, 0);
318
thd->version=refresh_version;
319
thd->security_ctx->priv_user=
320
thd->security_ctx->user= (char*) my_strdup("boot", MYF(MY_WME));
321
thd->security_ctx->priv_host[0]=0;
323
Make the "client" handle multiple results. This is necessary
324
to enable stored procedures with SELECTs and Dynamic SQL
327
thd->client_capabilities|= CLIENT_MULTI_RESULTS;
329
buff= (char*) thd->net.buff;
330
thd->init_for_queries();
331
while (fgets(buff, thd->net.max_packet, file))
333
/* strlen() can't be deleted because fgets() doesn't return length */
334
ulong length= (ulong) strlen(buff);
335
while (buff[length-1] != '\n' && !feof(file))
338
We got only a part of the current string. Will try to increase
339
net buffer then read the rest of the current string.
341
/* purecov: begin tested */
342
if (net_realloc(&(thd->net), 2 * thd->net.max_packet))
344
net_end_statement(thd);
348
buff= (char*) thd->net.buff;
349
fgets(buff + length, thd->net.max_packet - length, file);
350
length+= (ulong) strlen(buff + length);
354
break; /* purecov: inspected */
356
while (length && (my_isspace(thd->charset(), buff[length-1]) ||
357
buff[length-1] == ';'))
361
/* Skip lines starting with delimiter */
362
if (strncmp(buff, STRING_WITH_LEN("delimiter")) == 0)
365
thd->query_length=length;
366
thd->query= (char*) thd->memdup_w_gap(buff, length+1,
368
thd->query[length] = '\0';
369
DBUG_PRINT("query",("%-.4096s",thd->query));
372
We don't need to obtain LOCK_thread_count here because in bootstrap
373
mode we have only one thread.
375
thd->query_id=next_query_id();
377
mysql_parse(thd, thd->query, length, & found_semicolon);
378
close_thread_tables(thd); // Free tables
380
bootstrap_error= thd->is_error();
381
net_end_statement(thd);
386
free_root(thd->mem_root,MYF(MY_KEEP_PREALLOC));
387
free_root(&thd->transaction.mem_root,MYF(MY_KEEP_PREALLOC));
395
(void) pthread_mutex_lock(&LOCK_thread_count);
397
(void) pthread_mutex_unlock(&LOCK_thread_count);
398
(void) pthread_cond_broadcast(&COND_thread_count);
404
/* This works because items are allocated with sql_alloc() */
406
void free_items(Item *item)
409
DBUG_ENTER("free_items");
410
for (; item ; item=next)
418
/* This works because items are allocated with sql_alloc() */
420
void cleanup_items(Item *item)
422
DBUG_ENTER("cleanup_items");
423
for (; item ; item=item->next)
429
Ends the current transaction and (maybe) begin the next.
431
@param thd Current thread
432
@param completion Completion type
438
int end_trans(THD *thd, enum enum_mysql_completiontype completion)
442
DBUG_ENTER("end_trans");
444
if (unlikely(thd->in_sub_stmt))
446
my_error(ER_COMMIT_NOT_ALLOWED_IN_SF_OR_TRG, MYF(0));
449
if (thd->transaction.xid_state.xa_state != XA_NOTR)
451
my_error(ER_XAER_RMFAIL, MYF(0),
452
xa_state_names[thd->transaction.xid_state.xa_state]);
455
switch (completion) {
458
We don't use end_active_trans() here to ensure that this works
459
even if there is a problem with the OPTION_AUTO_COMMIT flag
460
(Which of course should never happen...)
462
thd->server_status&= ~SERVER_STATUS_IN_TRANS;
464
thd->options&= ~(OPTION_BEGIN | OPTION_KEEP_LOG);
465
thd->transaction.all.modified_non_trans_table= FALSE;
468
do_release= 1; /* fall through */
469
case COMMIT_AND_CHAIN:
470
res= end_active_trans(thd);
471
if (!res && completion == COMMIT_AND_CHAIN)
472
res= begin_trans(thd);
474
case ROLLBACK_RELEASE:
475
do_release= 1; /* fall through */
477
case ROLLBACK_AND_CHAIN:
479
thd->server_status&= ~SERVER_STATUS_IN_TRANS;
480
if (ha_rollback(thd))
482
thd->options&= ~(OPTION_BEGIN | OPTION_KEEP_LOG);
483
thd->transaction.all.modified_non_trans_table= FALSE;
484
if (!res && (completion == ROLLBACK_AND_CHAIN))
485
res= begin_trans(thd);
490
my_error(ER_UNKNOWN_COM_ERROR, MYF(0));
495
my_error(thd->killed_errno(), MYF(0));
496
else if ((res == 0) && do_release)
497
thd->killed= THD::KILL_CONNECTION;
504
Read one command from connection and execute it (query or simple command).
505
This function is called in loop from thread function.
507
For profiling to work, it must never be called recursively.
512
1 request of thread shutdown (see dispatch_command() description)
515
bool do_command(THD *thd)
521
enum enum_server_command command;
522
DBUG_ENTER("do_command");
525
indicator of uninitialized lex => normal flow of errors handling
528
thd->lex->current_select= 0;
531
This thread will do a blocking read from the client which
532
will be interrupted when the next command is received from
533
the client, the connection is closed or "net_wait_timeout"
534
number of seconds has passed
536
my_net_set_read_timeout(net, thd->variables.net_wait_timeout);
539
XXX: this code is here only to clear possible errors of init_connect.
540
Consider moving to init_connect() instead.
542
thd->clear_error(); // Clear error message
543
thd->main_da.reset_diagnostics_area();
545
net_new_transaction(net);
547
packet_length= my_net_read(net);
548
if (packet_length == packet_error)
550
DBUG_PRINT("info",("Got error %d reading command from socket %s",
552
vio_description(net->vio)));
554
/* Check if we can continue without closing the connection */
556
/* The error must be set. */
557
DBUG_ASSERT(thd->is_error());
558
net_end_statement(thd);
562
return_value= TRUE; // We have to close it.
571
packet= (char*) net->read_pos;
573
'packet_length' contains length of data, as it was stored in packet
574
header. In case of malformed header, my_net_read returns zero.
575
If packet_length is not zero, my_net_read ensures that the returned
576
number of bytes was actually read from network.
577
There is also an extra safety measure in my_net_read:
578
it sets packet[packet_length]= 0, but only for non-zero packets.
580
if (packet_length == 0) /* safety */
582
/* Initialize with COM_SLEEP packet */
583
packet[0]= (uchar) COM_SLEEP;
586
/* Do not rely on my_net_read, extra safety against programming errors. */
587
packet[packet_length]= '\0'; /* safety */
589
command= (enum enum_server_command) (uchar) packet[0];
591
if (command >= COM_END)
592
command= COM_END; // Wrong command
594
DBUG_PRINT("info",("Command on %s = %d (%s)",
595
vio_description(net->vio), command,
596
command_name[command].str));
598
/* Restore read timeout value */
599
my_net_set_read_timeout(net, thd->variables.net_read_timeout);
601
DBUG_ASSERT(packet_length);
602
return_value= dispatch_command(command, thd, packet+1, (uint) (packet_length-1));
605
DBUG_RETURN(return_value);
609
Determine if an attempt to update a non-temporary table while the
610
read-only option was enabled has been made.
612
This is a helper function to mysql_execute_command.
614
@note SQLCOM_MULTI_UPDATE is an exception and dealt with elsewhere.
616
@see mysql_execute_command
619
@retval TRUE The statement should be denied.
620
@retval FALSE The statement isn't updating any relevant tables.
623
static my_bool deny_updates_if_read_only_option(THD *thd,
624
TABLE_LIST *all_tables)
626
DBUG_ENTER("deny_updates_if_read_only_option");
633
if (!(sql_command_flags[lex->sql_command] & CF_CHANGES_DATA))
636
/* Multi update is an exception and is dealt with later. */
637
if (lex->sql_command == SQLCOM_UPDATE_MULTI)
640
const my_bool create_temp_tables=
641
(lex->sql_command == SQLCOM_CREATE_TABLE) &&
642
(lex->create_info.options & HA_LEX_CREATE_TMP_TABLE);
644
const my_bool drop_temp_tables=
645
(lex->sql_command == SQLCOM_DROP_TABLE) &&
648
const my_bool update_real_tables=
649
some_non_temp_table_to_be_updated(thd, all_tables) &&
650
!(create_temp_tables || drop_temp_tables);
653
const my_bool create_or_drop_databases=
654
(lex->sql_command == SQLCOM_CREATE_DB) ||
655
(lex->sql_command == SQLCOM_DROP_DB);
657
if (update_real_tables || create_or_drop_databases)
660
An attempt was made to modify one or more non-temporary tables.
666
/* Assuming that only temporary tables are modified. */
147
671
Perform one connection-level (COM_XXXX) command.
149
673
@param command type of command to perform
150
@param session connection handle
674
@param thd connection handle
151
675
@param packet data for the command, packet is always null-terminated
152
676
@param packet_length length of packet + 1 (to show that data is
153
677
null-terminated) except for COM_SLEEP, where it
157
set session->lex->sql_command to SQLCOM_END here.
681
set thd->lex->sql_command to SQLCOM_END here.
159
683
The following has to be changed to an 8 byte integer
164
688
1 request of thread shutdown, i. e. if command is
165
689
COM_QUIT/COM_SHUTDOWN
167
bool dispatch_command(enum enum_server_command command, Session *session,
168
char* packet, uint32_t packet_length)
691
bool dispatch_command(enum enum_server_command command, THD *thd,
692
char* packet, uint packet_length)
171
Query_id &query_id= Query_id::get_query_id();
173
DRIZZLE_COMMAND_START(session->thread_id,
176
session->command= command;
177
session->lex->sql_command= SQLCOM_END; /* to avoid confusing VIEW detectors */
179
session->setQueryId(query_id.value());
696
DBUG_ENTER("dispatch_command");
697
DBUG_PRINT("info",("packet: '%*.s'; command: %d", packet_length, packet, command));
699
thd->command=command;
701
Commands which always take a long time are logged into
702
the slow log only if opt_log_slow_admin_statements is set.
704
thd->enable_slow_log= TRUE;
705
thd->lex->sql_command= SQLCOM_END; /* to avoid confusing VIEW detectors */
707
VOID(pthread_mutex_lock(&LOCK_thread_count));
708
thd->query_id= global_query_id;
181
710
switch( command ) {
182
711
/* Ignore these statements. */
715
/* Only increase id on these statements but don't count them. */
716
case COM_STMT_PREPARE:
185
721
/* Increase id and count all other statements. */
187
statistic_increment(session->status_var.questions, &LOCK_status);
723
statistic_increment(thd->status_var.questions, &LOCK_status);
191
/* TODO: set session->lex->sql_command to SQLCOM_END here */
193
plugin::Logging::preDo(session);
195
session->server_status&=
728
/* TODO: set thd->lex->sql_command to SQLCOM_END here */
729
VOID(pthread_mutex_unlock(&LOCK_thread_count));
196
732
~(SERVER_QUERY_NO_INDEX_USED | SERVER_QUERY_NO_GOOD_INDEX_USED);
197
733
switch (command) {
198
734
case COM_INIT_DB:
201
status_var_increment(session->status_var.com_stat[SQLCOM_CHANGE_DB]);
203
tmp.length= packet_length;
204
if (!mysql_change_db(session, &tmp, false))
737
status_var_increment(thd->status_var.com_stat[SQLCOM_CHANGE_DB]);
738
thd->convert_string(&tmp, system_charset_info,
739
packet, packet_length, thd->charset());
740
if (!mysql_change_db(thd, &tmp, FALSE))
742
general_log_write(thd, command, thd->db, thd->db_length);
747
case COM_REGISTER_SLAVE:
749
if (!register_slave(thd, (uchar*)packet, packet_length))
753
case COM_CHANGE_USER:
755
status_var_increment(thd->status_var.com_other);
756
char *user= (char*) packet, *packet_end= packet + packet_length;
757
/* Safe because there is always a trailing \0 at the end of the packet */
758
char *passwd= strend(user)+1;
760
thd->clear_error(); // if errors from rollback
763
Old clients send null-terminated string ('\0' for empty string) for
764
password. New clients send the size (1 byte) + string (not null
765
terminated, so also '\0' for empty string).
767
Cast *passwd to an unsigned char, so that it doesn't extend the sign
768
for *passwd > 127 and become 2**32-127 after casting to uint.
770
char db_buff[NAME_LEN+1]; // buffer to store db in utf8
774
If there is no password supplied, the packet must contain '\0',
775
in any type of handshake (4.1 or pre-4.1).
777
if (passwd >= packet_end)
779
my_message(ER_UNKNOWN_COM_ERROR, ER(ER_UNKNOWN_COM_ERROR), MYF(0));
782
uint passwd_len= (thd->client_capabilities & CLIENT_SECURE_CONNECTION ?
783
(uchar)(*passwd++) : strlen(passwd));
784
uint dummy_errors, save_db_length, db_length;
786
Security_context save_security_ctx= *thd->security_ctx;
787
USER_CONN *save_user_connect;
791
Database name is always NUL-terminated, so in case of empty database
792
the packet must contain at least the trailing '\0'.
794
if (db >= packet_end)
796
my_message(ER_UNKNOWN_COM_ERROR, ER(ER_UNKNOWN_COM_ERROR), MYF(0));
799
db_length= strlen(db);
801
char *ptr= db + db_length + 1;
804
if (ptr < packet_end)
806
if (ptr + 2 > packet_end)
808
my_message(ER_UNKNOWN_COM_ERROR, ER(ER_UNKNOWN_COM_ERROR), MYF(0));
812
cs_number= uint2korr(ptr);
815
/* Convert database name to utf8 */
816
db_buff[copy_and_convert(db_buff, sizeof(db_buff)-1,
817
system_charset_info, db, db_length,
818
thd->charset(), &dummy_errors)]= 0;
821
/* Save user and privileges */
822
save_db_length= thd->db_length;
824
save_user_connect= thd->user_connect;
826
if (!(thd->security_ctx->user= my_strdup(user, MYF(0))))
828
thd->security_ctx->user= save_security_ctx.user;
829
my_message(ER_OUT_OF_RESOURCES, ER(ER_OUT_OF_RESOURCES), MYF(0));
833
/* Clear variables that are allocated */
834
thd->user_connect= 0;
835
thd->security_ctx->priv_user= thd->security_ctx->user;
836
res= check_user(thd, COM_CHANGE_USER, passwd, passwd_len, db, FALSE);
840
x_free(thd->security_ctx->user);
841
*thd->security_ctx= save_security_ctx;
842
thd->user_connect= save_user_connect;
844
thd->db_length= save_db_length;
849
x_free(save_security_ctx.user);
853
thd_init_client_charset(thd, cs_number);
854
thd->update_charset();
859
case COM_STMT_EXECUTE:
861
case COM_STMT_SEND_LONG_DATA:
862
case COM_STMT_PREPARE:
866
/* We should toss an error here */
212
if (! session->readAndStoreQuery(packet, packet_length))
871
if (alloc_query(thd, packet, packet_length))
213
872
break; // fatal error is set
214
DRIZZLE_QUERY_START(session->query.c_str(),
216
const_cast<const char *>(session->db.empty() ? "" : session->db.c_str()));
218
mysql_parse(session, session->query.c_str(), session->query.length());
873
char *packet_end= thd->query + thd->query_length;
874
/* 'b' stands for 'buffer' parameter', special for 'my_snprintf' */
875
const char* end_of_stmt= NULL;
877
general_log_write(thd, command, thd->query, thd->query_length);
878
DBUG_PRINT("query",("%-.4096s",thd->query));
880
if (!(specialflag & SPECIAL_NO_PRIOR))
881
my_pthread_setprio(pthread_self(),QUERY_PRIOR);
883
mysql_parse(thd, thd->query, thd->query_length, &end_of_stmt);
885
while (!thd->killed && (end_of_stmt != NULL) && ! thd->is_error())
887
char *beginning_of_next_stmt= (char*) end_of_stmt;
889
net_end_statement(thd);
891
Multiple queries exits, execute them individually
893
close_thread_tables(thd);
894
ulong length= (ulong)(packet_end - beginning_of_next_stmt);
896
log_slow_statement(thd);
898
/* Remove garbage at start of query */
899
while (length > 0 && my_isspace(thd->charset(), *beginning_of_next_stmt))
901
beginning_of_next_stmt++;
905
VOID(pthread_mutex_lock(&LOCK_thread_count));
906
thd->query_length= length;
907
thd->query= beginning_of_next_stmt;
909
Count each statement from the client.
911
statistic_increment(thd->status_var.questions, &LOCK_status);
912
thd->query_id= next_query_id();
913
thd->set_time(); /* Reset the query start time. */
914
/* TODO: set thd->lex->sql_command to SQLCOM_END here */
915
VOID(pthread_mutex_unlock(&LOCK_thread_count));
917
mysql_parse(thd, beginning_of_next_stmt, length, &end_of_stmt);
920
if (!(specialflag & SPECIAL_NO_PRIOR))
921
my_pthread_setprio(pthread_self(),WAIT_PRIOR);
922
DBUG_PRINT("info",("query ready"));
925
case COM_FIELD_LIST: // This isn't actually needed
927
char *fields, *packet_end= packet + packet_length, *arg_end;
928
/* Locked closure of all tables */
929
TABLE_LIST table_list;
930
LEX_STRING conv_name;
932
/* used as fields initializator */
935
status_var_increment(thd->status_var.com_stat[SQLCOM_SHOW_FIELDS]);
936
bzero((char*) &table_list,sizeof(table_list));
937
if (thd->copy_db_to(&table_list.db, &table_list.db_length))
940
We have name + wildcard in packet, separated by endzero
942
arg_end= strend(packet);
943
thd->convert_string(&conv_name, system_charset_info,
944
packet, (uint) (arg_end - packet), thd->charset());
945
table_list.alias= table_list.table_name= conv_name.str;
948
if (!my_strcasecmp(system_charset_info, table_list.db,
949
INFORMATION_SCHEMA_NAME.str))
951
ST_SCHEMA_TABLE *schema_table= find_schema_table(thd, table_list.alias);
953
table_list.schema_table= schema_table;
956
thd->query_length= (uint) (packet_end - packet); // Don't count end \0
957
if (!(thd->query=fields= (char*) thd->memdup(packet,thd->query_length+1)))
959
general_log_print(thd, command, "%s %s", table_list.table_name, fields);
960
if (lower_case_table_names)
961
my_casedn_str(files_charset_info, table_list.table_name);
963
/* init structures for VIEW processing */
964
table_list.select_lex= &(thd->lex->select_lex);
967
mysql_reset_thd_for_next_command(thd);
970
select_lex.table_list.link_in_list((uchar*) &table_list,
971
(uchar**) &table_list.next_local);
972
thd->lex->add_to_query_tables(&table_list);
974
/* switch on VIEW optimisation: do not fill temporary tables */
975
thd->lex->sql_command= SQLCOM_SHOW_FIELDS;
976
mysqld_list_fields(thd,&table_list,fields);
977
thd->lex->unit.cleanup();
978
thd->cleanup_after_query();
223
982
/* We don't calculate statistics for this command */
224
session->main_da.disable_status(); // Don't send anything back
225
error=true; // End server
983
general_log_print(thd, command, NullS);
984
net->error=0; // Don't give 'abort' message
985
thd->main_da.disable_status(); // Don't send anything back
986
error=TRUE; // End server
988
case COM_BINLOG_DUMP:
992
uint32 slave_server_id;
994
status_var_increment(thd->status_var.com_other);
995
thd->enable_slow_log= opt_log_slow_admin_statements;
996
/* TODO: The following has to be changed to an 8 byte integer */
997
pos = uint4korr(packet);
998
flags = uint2korr(packet + 4);
999
thd->server_id=0; /* avoid suicide */
1000
if ((slave_server_id= uint4korr(packet+6))) // mysqlbinlog.server_id==0
1001
kill_zombie_dump_threads(slave_server_id);
1002
thd->server_id = slave_server_id;
1004
general_log_print(thd, command, "Log: '%s' Pos: %ld", packet+10,
1006
mysql_binlog_send(thd, thd->strdup(packet + 10), (my_off_t) pos, flags);
1007
unregister_slave(thd,1,1);
1008
/* fake COM_QUIT -- if we get here, the thread needs to terminate */
227
1012
case COM_SHUTDOWN:
229
status_var_increment(session->status_var.com_other);
231
session->close_thread_tables(); // Free before kill
1014
status_var_increment(thd->status_var.com_other);
1016
If the client is < 4.1.3, it is going to send us no argument; then
1017
packet_length is 0, packet[0] is the end 0 of the packet. Note that
1018
SHUTDOWN_DEFAULT is 0. If client is >= 4.1.3, the shutdown level is in
1021
enum mysql_enum_shutdown_level level=
1022
(enum mysql_enum_shutdown_level) (uchar) packet[0];
1023
if (level == SHUTDOWN_DEFAULT)
1024
level= SHUTDOWN_WAIT_ALL_BUFFERS; // soon default will be configurable
1025
else if (level != SHUTDOWN_WAIT_ALL_BUFFERS)
1027
my_error(ER_NOT_SUPPORTED_YET, MYF(0), "this shutdown level");
1030
DBUG_PRINT("quit",("Got shutdown command for level %u", level));
1031
general_log_print(thd, command, NullS);
1033
close_thread_tables(thd); // Free before kill
1038
case COM_STATISTICS:
1040
STATUS_VAR current_global_status_var;
1043
ulonglong queries_per_second1000;
1045
uint buff_len= sizeof(buff);
1047
general_log_print(thd, command, NullS);
1048
status_var_increment(thd->status_var.com_stat[SQLCOM_SHOW_STATUS]);
1049
calc_sum_of_all_status(¤t_global_status_var);
1050
if (!(uptime= (ulong) (thd->start_time - server_start_time)))
1051
queries_per_second1000= 0;
1053
queries_per_second1000= thd->query_id * LL(1000) / uptime;
1055
length= my_snprintf((char*) buff, buff_len - 1,
1056
"Uptime: %lu Threads: %d Questions: %lu "
1057
"Slow queries: %lu Opens: %lu Flush tables: %lu "
1058
"Open tables: %u Queries per second avg: %u.%u",
1060
(int) thread_count, (ulong) thd->query_id,
1061
current_global_status_var.long_query_count,
1062
current_global_status_var.opened_tables,
1064
cached_open_tables(),
1065
(uint) (queries_per_second1000 / 1000),
1066
(uint) (queries_per_second1000 % 1000));
1067
/* Store the buffer in permanent memory */
1068
my_ok(thd, 0, 0, buff);
1069
VOID(my_net_write(net, (uchar*) buff, length));
1070
VOID(net_flush(net));
1071
thd->main_da.disable_status();
237
status_var_increment(session->status_var.com_other);
238
session->my_ok(); // Tell client we are alive
1075
status_var_increment(thd->status_var.com_other);
1076
my_ok(thd); // Tell client we are alive
1078
case COM_PROCESS_INFO:
1079
status_var_increment(thd->status_var.com_stat[SQLCOM_SHOW_PROCESSLIST]);
1080
general_log_print(thd, command, NullS);
1081
mysqld_list_processes(thd, NullS, 0);
1083
case COM_PROCESS_KILL:
1085
status_var_increment(thd->status_var.com_stat[SQLCOM_KILL]);
1086
ulong id=(ulong) uint4korr(packet);
1087
sql_kill(thd,id,false);
1090
case COM_SET_OPTION:
1092
status_var_increment(thd->status_var.com_stat[SQLCOM_SET_OPTION]);
1093
uint opt_command= uint2korr(packet);
1095
switch (opt_command) {
1096
case (int) MYSQL_OPTION_MULTI_STATEMENTS_ON:
1097
thd->client_capabilities|= CLIENT_MULTI_STATEMENTS;
1100
case (int) MYSQL_OPTION_MULTI_STATEMENTS_OFF:
1101
thd->client_capabilities&= ~CLIENT_MULTI_STATEMENTS;
1105
my_message(ER_UNKNOWN_COM_ERROR, ER(ER_UNKNOWN_COM_ERROR), MYF(0));
1111
status_var_increment(thd->status_var.com_other);
1112
mysql_print_status();
1113
general_log_print(thd, command, NullS);
241
1117
case COM_CONNECT: // Impossible here
1118
case COM_TIME: // Impossible from client
244
1121
my_message(ER_UNKNOWN_COM_ERROR, ER(ER_UNKNOWN_COM_ERROR), MYF(0));
496
1440
variables, but for now this is probably good enough.
497
1441
Don't reset warnings when executing a stored routine.
499
if (all_tables || ! lex->is_single_level_stmt())
501
drizzle_reset_errors(session, 0);
504
status_var_increment(session->status_var.com_stat[lex->sql_command]);
506
assert(session->transaction.stmt.hasModifiedNonTransData() == false);
508
/* now we are ready to execute the statement */
509
res= lex->statement->execute();
511
session->set_proc_info("query end");
1443
if (all_tables || !lex->is_single_level_stmt())
1444
mysql_reset_errors(thd, 0);
1446
#ifdef HAVE_REPLICATION
1447
if (unlikely(thd->slave_thread))
1450
Check if statment should be skipped because of slave filtering
1454
- UPDATE MULTI: For this statement, we want to check the filtering
1455
rules later in the code
1456
- SET: we always execute it (Not that many SET commands exists in
1457
the binary log anyway -- only 4.1 masters write SET statements,
1458
in 5.0 there are no SET statements in the binary log)
1459
- DROP TEMPORARY TABLE IF EXISTS: we always execute it (otherwise we
1460
have stale files on slave caused by exclusion of one tmp table).
1462
if (!(lex->sql_command == SQLCOM_UPDATE_MULTI) &&
1463
!(lex->sql_command == SQLCOM_SET_OPTION) &&
1464
!(lex->sql_command == SQLCOM_DROP_TABLE &&
1465
lex->drop_temporary && lex->drop_if_exists) &&
1466
all_tables_not_ok(thd, all_tables))
1468
/* we warn the slave SQL thread */
1469
my_message(ER_SLAVE_IGNORED_TABLE, ER(ER_SLAVE_IGNORED_TABLE), MYF(0));
1470
if (thd->one_shot_set)
1473
It's ok to check thd->one_shot_set here:
1475
The charsets in a MySQL 5.0 slave can change by both a binlogged
1476
SET ONE_SHOT statement and the event-internal charset setting,
1477
and these two ways to change charsets do not seems to work
1480
At least there seems to be problems in the rli cache for
1481
charsets if we are using ONE_SHOT. Note that this is normally no
1482
problem because either the >= 5.0 slave reads a 4.1 binlog (with
1483
ONE_SHOT) *or* or 5.0 binlog (without ONE_SHOT) but never both."
1485
reset_one_shot_variables(thd);
1492
#endif /* HAVE_REPLICATION */
1494
When option readonly is set deny operations which change non-temporary
1495
tables. Except for the replication thread and the 'super' users.
1497
if (deny_updates_if_read_only_option(thd, all_tables))
1499
my_error(ER_OPTION_PREVENTS_STATEMENT, MYF(0), "--read-only");
1502
#ifdef HAVE_REPLICATION
1503
} /* endif unlikely slave */
1505
status_var_increment(thd->status_var.com_stat[lex->sql_command]);
1507
DBUG_ASSERT(thd->transaction.stmt.modified_non_trans_table == FALSE);
1509
switch (lex->sql_command) {
1510
case SQLCOM_SHOW_STATUS:
1512
system_status_var old_status_var= thd->status_var;
1513
thd->initial_status_var= &old_status_var;
1514
res= execute_sqlcom_select(thd, all_tables);
1515
/* Don't log SHOW STATUS commands to slow query log */
1516
thd->server_status&= ~(SERVER_QUERY_NO_INDEX_USED |
1517
SERVER_QUERY_NO_GOOD_INDEX_USED);
1519
restore status variables, as we don't want 'show status' to cause
1522
pthread_mutex_lock(&LOCK_status);
1523
add_diff_to_status(&global_status_var, &thd->status_var,
1525
thd->status_var= old_status_var;
1526
pthread_mutex_unlock(&LOCK_status);
1529
case SQLCOM_SHOW_DATABASES:
1530
case SQLCOM_SHOW_TABLES:
1531
case SQLCOM_SHOW_TABLE_STATUS:
1532
case SQLCOM_SHOW_OPEN_TABLES:
1533
case SQLCOM_SHOW_FIELDS:
1534
case SQLCOM_SHOW_KEYS:
1535
case SQLCOM_SHOW_VARIABLES:
1536
case SQLCOM_SHOW_CHARSETS:
1537
case SQLCOM_SHOW_COLLATIONS:
1540
thd->status_var.last_query_cost= 0.0;
1541
res= execute_sqlcom_select(thd, all_tables);
1544
case SQLCOM_PREPARE:
1548
case SQLCOM_EXECUTE:
1552
case SQLCOM_DEALLOCATE_PREPARE:
1556
case SQLCOM_EMPTY_QUERY:
1562
res = purge_master_logs(thd, lex->to_log);
1565
case SQLCOM_PURGE_BEFORE:
1569
/* PURGE MASTER LOGS BEFORE 'data' */
1570
it= (Item *)lex->value_list.head();
1571
if ((!it->fixed && it->fix_fields(lex->thd, &it)) ||
1574
my_error(ER_WRONG_ARGUMENTS, MYF(0), "PURGE LOGS BEFORE");
1577
it= new Item_func_unix_timestamp(it);
1579
it is OK only emulate fix_fieds, because we need only
1582
it->quick_fix_field();
1583
res = purge_master_logs_before_date(thd, (ulong)it->val_int());
1586
case SQLCOM_SHOW_WARNS:
1588
res= mysqld_show_warnings(thd, (ulong)
1589
((1L << (uint) MYSQL_ERROR::WARN_LEVEL_NOTE) |
1590
(1L << (uint) MYSQL_ERROR::WARN_LEVEL_WARN) |
1591
(1L << (uint) MYSQL_ERROR::WARN_LEVEL_ERROR)
1595
case SQLCOM_SHOW_ERRORS:
1597
res= mysqld_show_warnings(thd, (ulong)
1598
(1L << (uint) MYSQL_ERROR::WARN_LEVEL_ERROR));
1601
case SQLCOM_SHOW_NEW_MASTER:
1603
/* This query don't work now. See comment in repl_failsafe.cc */
1604
#ifndef WORKING_NEW_MASTER
1605
my_error(ER_NOT_SUPPORTED_YET, MYF(0), "SHOW NEW MASTER");
1608
res = show_new_master(thd);
1613
#ifdef HAVE_REPLICATION
1614
case SQLCOM_SHOW_SLAVE_HOSTS:
1616
res = show_slave_hosts(thd);
1619
case SQLCOM_SHOW_BINLOG_EVENTS:
1621
res = mysql_show_binlog_events(thd);
1626
case SQLCOM_ASSIGN_TO_KEYCACHE:
1628
DBUG_ASSERT(first_table == all_tables && first_table != 0);
1629
res= mysql_assign_to_keycache(thd, first_table, &lex->ident);
1632
case SQLCOM_PRELOAD_KEYS:
1634
DBUG_ASSERT(first_table == all_tables && first_table != 0);
1635
res = mysql_preload_keys(thd, first_table);
1638
#ifdef HAVE_REPLICATION
1639
case SQLCOM_CHANGE_MASTER:
1641
pthread_mutex_lock(&LOCK_active_mi);
1642
res = change_master(thd,active_mi);
1643
pthread_mutex_unlock(&LOCK_active_mi);
1646
case SQLCOM_SHOW_SLAVE_STAT:
1648
pthread_mutex_lock(&LOCK_active_mi);
1649
if (active_mi != NULL)
1651
res = show_master_info(thd, active_mi);
1655
push_warning(thd, MYSQL_ERROR::WARN_LEVEL_WARN, 0,
1656
"the master info structure does not exist");
1659
pthread_mutex_unlock(&LOCK_active_mi);
1662
case SQLCOM_SHOW_MASTER_STAT:
1664
res = show_binlog_info(thd);
1668
#endif /* HAVE_REPLICATION */
1669
case SQLCOM_SHOW_ENGINE_STATUS:
1671
res = ha_show_status(thd, lex->create_info.db_type, HA_ENGINE_STATUS);
1674
case SQLCOM_SHOW_ENGINE_MUTEX:
1676
res = ha_show_status(thd, lex->create_info.db_type, HA_ENGINE_MUTEX);
1679
case SQLCOM_CREATE_TABLE:
1681
/* If CREATE TABLE of non-temporary table, do implicit commit */
1682
if (!(lex->create_info.options & HA_LEX_CREATE_TMP_TABLE))
1684
if (end_active_trans(thd))
1690
DBUG_ASSERT(first_table == all_tables && first_table != 0);
1692
// Skip first table, which is the table we are creating
1693
TABLE_LIST *create_table= lex->unlink_first_table(&link_to_local);
1694
TABLE_LIST *select_tables= lex->query_tables;
1696
Code below (especially in mysql_create_table() and select_create
1697
methods) may modify HA_CREATE_INFO structure in LEX, so we have to
1698
use a copy of this structure to make execution prepared statement-
1699
safe. A shallow copy is enough as this code won't modify any memory
1700
referenced from this structure.
1702
HA_CREATE_INFO create_info(lex->create_info);
1704
We need to copy alter_info for the same reasons of re-execution
1705
safety, only in case of Alter_info we have to do (almost) a deep
1708
Alter_info alter_info(lex->alter_info, thd->mem_root);
1710
if (thd->is_fatal_error)
1712
/* If out of memory when creating a copy of alter_info. */
1714
goto end_with_restore_list;
1717
if ((res= create_table_precheck(thd, select_tables, create_table)))
1718
goto end_with_restore_list;
1720
/* Might have been updated in create_table_precheck */
1721
create_info.alias= create_table->alias;
1723
#ifdef HAVE_READLINK
1724
/* Fix names if symlinked tables */
1725
if (append_file_to_dir(thd, &create_info.data_file_name,
1726
create_table->table_name) ||
1727
append_file_to_dir(thd, &create_info.index_file_name,
1728
create_table->table_name))
1729
goto end_with_restore_list;
1732
If we are using SET CHARSET without DEFAULT, add an implicit
1733
DEFAULT to not confuse old users. (This may change).
1735
if ((create_info.used_fields &
1736
(HA_CREATE_USED_DEFAULT_CHARSET | HA_CREATE_USED_CHARSET)) ==
1737
HA_CREATE_USED_CHARSET)
1739
create_info.used_fields&= ~HA_CREATE_USED_CHARSET;
1740
create_info.used_fields|= HA_CREATE_USED_DEFAULT_CHARSET;
1741
create_info.default_table_charset= create_info.table_charset;
1742
create_info.table_charset= 0;
1745
The create-select command will open and read-lock the select table
1746
and then create, open and write-lock the new table. If a global
1747
read lock steps in, we get a deadlock. The write lock waits for
1748
the global read lock, while the global read lock waits for the
1749
select table to be closed. So we wait until the global readlock is
1750
gone before starting both steps. Note that
1751
wait_if_global_read_lock() sets a protection against a new global
1752
read lock when it succeeds. This needs to be released by
1753
start_waiting_global_read_lock(). We protect the normal CREATE
1754
TABLE in the same way. That way we avoid that a new table is
1755
created during a gobal read lock.
1757
if (!thd->locked_tables &&
1758
!(need_start_waiting= !wait_if_global_read_lock(thd, 0, 1)))
1761
goto end_with_restore_list;
1763
if (select_lex->item_list.elements) // With select
1765
select_result *result;
1767
select_lex->options|= SELECT_NO_UNLOCK;
1768
unit->set_limit(select_lex);
1771
Disable non-empty MERGE tables with CREATE...SELECT. Too
1772
complicated. See Bug #26379. Empty MERGE tables are read-only
1773
and don't allow CREATE...SELECT anyway.
1775
if (create_info.used_fields & HA_CREATE_USED_UNION)
1777
my_error(ER_WRONG_OBJECT, MYF(0), create_table->db,
1778
create_table->table_name, "BASE TABLE");
1780
goto end_with_restore_list;
1783
if (!(create_info.options & HA_LEX_CREATE_TMP_TABLE))
1785
lex->link_first_table_back(create_table, link_to_local);
1786
create_table->create= TRUE;
1789
if (!(res= open_and_lock_tables(thd, lex->query_tables)))
1792
Is table which we are changing used somewhere in other parts
1795
if (!(create_info.options & HA_LEX_CREATE_TMP_TABLE))
1797
TABLE_LIST *duplicate;
1798
create_table= lex->unlink_first_table(&link_to_local);
1799
if ((duplicate= unique_table(thd, create_table, select_tables, 0)))
1801
update_non_unique_table_error(create_table, "CREATE", duplicate);
1803
goto end_with_restore_list;
1806
/* If we create merge table, we have to test tables in merge, too */
1807
if (create_info.used_fields & HA_CREATE_USED_UNION)
1810
for (tab= (TABLE_LIST*) create_info.merge_list.first;
1812
tab= tab->next_local)
1814
TABLE_LIST *duplicate;
1815
if ((duplicate= unique_table(thd, tab, select_tables, 0)))
1817
update_non_unique_table_error(tab, "CREATE", duplicate);
1819
goto end_with_restore_list;
1825
select_create is currently not re-execution friendly and
1826
needs to be created for every execution of a PS/SP.
1828
if ((result= new select_create(create_table,
1831
select_lex->item_list,
1837
CREATE from SELECT give its SELECT_LEX for SELECT,
1838
and item_list belong to SELECT
1840
res= handle_select(thd, lex, result, 0);
1844
else if (!(create_info.options & HA_LEX_CREATE_TMP_TABLE))
1845
create_table= lex->unlink_first_table(&link_to_local);
1850
/* So that CREATE TEMPORARY TABLE gets to binlog at commit/rollback */
1851
if (create_info.options & HA_LEX_CREATE_TMP_TABLE)
1852
thd->options|= OPTION_KEEP_LOG;
1853
/* regular create */
1854
if (create_info.options & HA_LEX_CREATE_TABLE_LIKE)
1855
res= mysql_create_like_table(thd, create_table, select_tables,
1859
res= mysql_create_table(thd, create_table->db,
1860
create_table->table_name, &create_info,
1867
/* put tables back for PS rexecuting */
1868
end_with_restore_list:
1869
lex->link_first_table_back(create_table, link_to_local);
1872
case SQLCOM_CREATE_INDEX:
1874
case SQLCOM_DROP_INDEX:
1876
CREATE INDEX and DROP INDEX are implemented by calling ALTER
1877
TABLE with proper arguments.
1879
In the future ALTER TABLE will notice that the request is to
1880
only add indexes and create these one by one for the existing
1881
table without having to do a full rebuild.
1884
/* Prepare stack copies to be re-execution safe */
1885
HA_CREATE_INFO create_info;
1886
Alter_info alter_info(lex->alter_info, thd->mem_root);
1888
if (thd->is_fatal_error) /* out of memory creating a copy of alter_info */
1891
DBUG_ASSERT(first_table == all_tables && first_table != 0);
1892
if (end_active_trans(thd))
1895
Currently CREATE INDEX or DROP INDEX cause a full table rebuild
1896
and thus classify as slow administrative statements just like
1899
thd->enable_slow_log= opt_log_slow_admin_statements;
1901
bzero((char*) &create_info, sizeof(create_info));
1902
create_info.db_type= 0;
1903
create_info.row_type= ROW_TYPE_NOT_USED;
1904
create_info.default_table_charset= thd->variables.collation_database;
1906
res= mysql_alter_table(thd, first_table->db, first_table->table_name,
1907
&create_info, first_table, &alter_info,
1911
#ifdef HAVE_REPLICATION
1912
case SQLCOM_SLAVE_START:
1914
pthread_mutex_lock(&LOCK_active_mi);
1915
start_slave(thd,active_mi,1 /* net report*/);
1916
pthread_mutex_unlock(&LOCK_active_mi);
1919
case SQLCOM_SLAVE_STOP:
1921
If the client thread has locked tables, a deadlock is possible.
1923
- the client thread does LOCK TABLE t READ.
1924
- then the master updates t.
1925
- then the SQL slave thread wants to update t,
1926
so it waits for the client thread because t is locked by it.
1927
- then the client thread does SLAVE STOP.
1928
SLAVE STOP waits for the SQL slave thread to terminate its
1929
update t, which waits for the client thread because t is locked by it.
1930
To prevent that, refuse SLAVE STOP if the
1931
client thread has locked tables
1933
if (thd->locked_tables || thd->active_transaction() || thd->global_read_lock)
1935
my_message(ER_LOCK_OR_ACTIVE_TRANSACTION,
1936
ER(ER_LOCK_OR_ACTIVE_TRANSACTION), MYF(0));
1940
pthread_mutex_lock(&LOCK_active_mi);
1941
stop_slave(thd,active_mi,1/* net report*/);
1942
pthread_mutex_unlock(&LOCK_active_mi);
1945
#endif /* HAVE_REPLICATION */
1947
case SQLCOM_ALTER_TABLE:
1948
DBUG_ASSERT(first_table == all_tables && first_table != 0);
1951
Code in mysql_alter_table() may modify its HA_CREATE_INFO argument,
1952
so we have to use a copy of this structure to make execution
1953
prepared statement- safe. A shallow copy is enough as no memory
1954
referenced from this structure will be modified.
1956
HA_CREATE_INFO create_info(lex->create_info);
1957
Alter_info alter_info(lex->alter_info, thd->mem_root);
1959
if (thd->is_fatal_error) /* out of memory creating a copy of alter_info */
1964
/* Must be set in the parser */
1965
DBUG_ASSERT(select_lex->db);
1967
{ // Rename of table
1968
TABLE_LIST tmp_table;
1969
bzero((char*) &tmp_table,sizeof(tmp_table));
1970
tmp_table.table_name= lex->name.str;
1971
tmp_table.db=select_lex->db;
1974
/* Don't yet allow changing of symlinks with ALTER TABLE */
1975
if (create_info.data_file_name)
1976
push_warning(thd, MYSQL_ERROR::WARN_LEVEL_WARN, 0,
1977
"DATA DIRECTORY option ignored");
1978
if (create_info.index_file_name)
1979
push_warning(thd, MYSQL_ERROR::WARN_LEVEL_WARN, 0,
1980
"INDEX DIRECTORY option ignored");
1981
create_info.data_file_name= create_info.index_file_name= NULL;
1982
/* ALTER TABLE ends previous transaction */
1983
if (end_active_trans(thd))
1986
if (!thd->locked_tables &&
1987
!(need_start_waiting= !wait_if_global_read_lock(thd, 0, 1)))
1993
thd->enable_slow_log= opt_log_slow_admin_statements;
1994
res= mysql_alter_table(thd, select_lex->db, lex->name.str,
1998
select_lex->order_list.elements,
1999
(ORDER *) select_lex->order_list.first,
2003
case SQLCOM_RENAME_TABLE:
2005
DBUG_ASSERT(first_table == all_tables && first_table != 0);
2007
for (table= first_table; table; table= table->next_local->next_local)
2009
TABLE_LIST old_list, new_list;
2011
we do not need initialize old_list and new_list because we will
2012
come table[0] and table->next[0] there
2015
new_list= table->next_local[0];
2018
if (end_active_trans(thd) || mysql_rename_tables(thd, first_table, 0))
2024
case SQLCOM_SHOW_BINLOGS:
2026
res = show_binlogs(thd);
2029
case SQLCOM_SHOW_CREATE:
2030
DBUG_ASSERT(first_table == all_tables && first_table != 0);
2032
res= mysqld_show_create(thd, first_table);
2035
case SQLCOM_CHECKSUM:
2037
DBUG_ASSERT(first_table == all_tables && first_table != 0);
2038
res = mysql_checksum_table(thd, first_table, &lex->check_opt);
2043
DBUG_ASSERT(first_table == all_tables && first_table != 0);
2044
thd->enable_slow_log= opt_log_slow_admin_statements;
2045
res= mysql_repair_table(thd, first_table, &lex->check_opt);
2046
/* ! we write after unlocking the table */
2047
if (!res && !lex->no_write_to_binlog)
2050
Presumably, REPAIR and binlog writing doesn't require synchronization
2052
write_bin_log(thd, TRUE, thd->query, thd->query_length);
2054
select_lex->table_list.first= (uchar*) first_table;
2055
lex->query_tables=all_tables;
2060
DBUG_ASSERT(first_table == all_tables && first_table != 0);
2061
thd->enable_slow_log= opt_log_slow_admin_statements;
2062
res = mysql_check_table(thd, first_table, &lex->check_opt);
2063
select_lex->table_list.first= (uchar*) first_table;
2064
lex->query_tables=all_tables;
2067
case SQLCOM_ANALYZE:
2069
DBUG_ASSERT(first_table == all_tables && first_table != 0);
2070
thd->enable_slow_log= opt_log_slow_admin_statements;
2071
res= mysql_analyze_table(thd, first_table, &lex->check_opt);
2072
/* ! we write after unlocking the table */
2073
if (!res && !lex->no_write_to_binlog)
2076
Presumably, ANALYZE and binlog writing doesn't require synchronization
2078
write_bin_log(thd, TRUE, thd->query, thd->query_length);
2080
select_lex->table_list.first= (uchar*) first_table;
2081
lex->query_tables=all_tables;
2085
case SQLCOM_OPTIMIZE:
2087
DBUG_ASSERT(first_table == all_tables && first_table != 0);
2088
thd->enable_slow_log= opt_log_slow_admin_statements;
2089
res= (specialflag & (SPECIAL_SAFE_MODE | SPECIAL_NO_NEW_FUNC)) ?
2090
mysql_recreate_table(thd, first_table) :
2091
mysql_optimize_table(thd, first_table, &lex->check_opt);
2092
/* ! we write after unlocking the table */
2093
if (!res && !lex->no_write_to_binlog)
2096
Presumably, OPTIMIZE and binlog writing doesn't require synchronization
2098
write_bin_log(thd, TRUE, thd->query, thd->query_length);
2100
select_lex->table_list.first= (uchar*) first_table;
2101
lex->query_tables=all_tables;
2105
DBUG_ASSERT(first_table == all_tables && first_table != 0);
2106
if (update_precheck(thd, all_tables))
2108
DBUG_ASSERT(select_lex->offset_limit == 0);
2109
unit->set_limit(select_lex);
2110
res= (up_result= mysql_update(thd, all_tables,
2111
select_lex->item_list,
2114
select_lex->order_list.elements,
2115
(ORDER *) select_lex->order_list.first,
2116
unit->select_limit_cnt,
2117
lex->duplicates, lex->ignore));
2118
/* mysql_update return 2 if we need to switch to multi-update */
2122
case SQLCOM_UPDATE_MULTI:
2124
DBUG_ASSERT(first_table == all_tables && first_table != 0);
2125
/* if we switched from normal update, rights are checked */
2128
if ((res= multi_update_precheck(thd, all_tables)))
2134
res= mysql_multi_update_prepare(thd);
2136
#ifdef HAVE_REPLICATION
2137
/* Check slave filtering rules */
2138
if (unlikely(thd->slave_thread))
2140
if (all_tables_not_ok(thd, all_tables))
2144
res= 0; /* don't care of prev failure */
2145
thd->clear_error(); /* filters are of highest prior */
2147
/* we warn the slave SQL thread */
2148
my_error(ER_SLAVE_IGNORED_TABLE, MYF(0));
2156
#endif /* HAVE_REPLICATION */
2160
some_non_temp_table_to_be_updated(thd, all_tables))
2162
my_error(ER_OPTION_PREVENTS_STATEMENT, MYF(0), "--read-only");
2165
#ifdef HAVE_REPLICATION
2169
res= mysql_multi_update(thd, all_tables,
2170
&select_lex->item_list,
2173
select_lex->options,
2174
lex->duplicates, lex->ignore, unit, select_lex);
2177
case SQLCOM_REPLACE:
2179
if (mysql_bin_log.is_open())
2182
Generate an incident log event before writing the real event
2183
to the binary log. We put this event is before the statement
2184
since that makes it simpler to check that the statement was
2185
not executed on the slave (since incidents usually stop the
2188
Observe that any row events that are generated will be
2191
This is only for testing purposes and will not be present in a
2195
Incident incident= INCIDENT_NONE;
2196
DBUG_PRINT("debug", ("Just before generate_incident()"));
2197
DBUG_EXECUTE_IF("incident_database_resync_on_replace",
2198
incident= INCIDENT_LOST_EVENTS;);
2201
Incident_log_event ev(thd, incident);
2202
mysql_bin_log.write(&ev);
2203
mysql_bin_log.rotate_and_purge(RP_FORCE_ROTATE);
2205
DBUG_PRINT("debug", ("Just after generate_incident()"));
2210
DBUG_ASSERT(first_table == all_tables && first_table != 0);
2211
if ((res= insert_precheck(thd, all_tables)))
2214
if (!thd->locked_tables &&
2215
!(need_start_waiting= !wait_if_global_read_lock(thd, 0, 1)))
2221
res= mysql_insert(thd, all_tables, lex->field_list, lex->many_values,
2222
lex->update_list, lex->value_list,
2223
lex->duplicates, lex->ignore);
2227
case SQLCOM_REPLACE_SELECT:
2228
case SQLCOM_INSERT_SELECT:
2230
select_result *sel_result;
2231
DBUG_ASSERT(first_table == all_tables && first_table != 0);
2232
if ((res= insert_precheck(thd, all_tables)))
2235
/* Fix lock for first table */
2236
if (first_table->lock_type == TL_WRITE_DELAYED)
2237
first_table->lock_type= TL_WRITE;
2239
/* Don't unlock tables until command is written to binary log */
2240
select_lex->options|= SELECT_NO_UNLOCK;
2242
unit->set_limit(select_lex);
2244
if (! thd->locked_tables &&
2245
! (need_start_waiting= ! wait_if_global_read_lock(thd, 0, 1)))
2251
if (!(res= open_and_lock_tables(thd, all_tables)))
2253
/* Skip first table, which is the table we are inserting in */
2254
TABLE_LIST *second_table= first_table->next_local;
2255
select_lex->table_list.first= (uchar*) second_table;
2256
select_lex->context.table_list=
2257
select_lex->context.first_name_resolution_table= second_table;
2258
res= mysql_insert_select_prepare(thd);
2259
if (!res && (sel_result= new select_insert(first_table,
2267
res= handle_select(thd, lex, sel_result, OPTION_SETUP_TABLES_DONE);
2269
Invalidate the table in the query cache if something changed
2270
after unlocking when changes become visible.
2271
TODO: this is workaround. right way will be move invalidating in
2272
the unlock procedure.
2274
if (first_table->lock_type == TL_WRITE_CONCURRENT_INSERT &&
2277
/* INSERT ... SELECT should invalidate only the very first table */
2278
TABLE_LIST *save_table= first_table->next_local;
2279
first_table->next_local= 0;
2280
first_table->next_local= save_table;
2284
/* revert changes for SP */
2285
select_lex->table_list.first= (uchar*) first_table;
2290
case SQLCOM_TRUNCATE:
2291
if (end_active_trans(thd))
2296
DBUG_ASSERT(first_table == all_tables && first_table != 0);
2298
Don't allow this within a transaction because we want to use
2301
if (thd->locked_tables || thd->active_transaction())
2303
my_message(ER_LOCK_OR_ACTIVE_TRANSACTION,
2304
ER(ER_LOCK_OR_ACTIVE_TRANSACTION), MYF(0));
2308
res= mysql_truncate(thd, first_table, 0);
2313
DBUG_ASSERT(first_table == all_tables && first_table != 0);
2314
DBUG_ASSERT(select_lex->offset_limit == 0);
2315
unit->set_limit(select_lex);
2317
if (!thd->locked_tables &&
2318
!(need_start_waiting= !wait_if_global_read_lock(thd, 0, 1)))
2324
res = mysql_delete(thd, all_tables, select_lex->where,
2325
&select_lex->order_list,
2326
unit->select_limit_cnt, select_lex->options,
2330
case SQLCOM_DELETE_MULTI:
2332
DBUG_ASSERT(first_table == all_tables && first_table != 0);
2333
TABLE_LIST *aux_tables=
2334
(TABLE_LIST *)thd->lex->auxiliary_table_list.first;
2335
multi_delete *del_result;
2337
if (!thd->locked_tables &&
2338
!(need_start_waiting= !wait_if_global_read_lock(thd, 0, 1)))
2344
if ((res= multi_delete_precheck(thd, all_tables)))
2347
/* condition will be TRUE on SP re-excuting */
2348
if (select_lex->item_list.elements != 0)
2349
select_lex->item_list.empty();
2350
if (add_item_to_list(thd, new Item_null()))
2353
thd_proc_info(thd, "init");
2354
if ((res= open_and_lock_tables(thd, all_tables)))
2357
if ((res= mysql_multi_delete_prepare(thd)))
2360
if (!thd->is_fatal_error &&
2361
(del_result= new multi_delete(aux_tables, lex->table_count)))
2363
res= mysql_select(thd, &select_lex->ref_pointer_array,
2364
select_lex->get_table_list(),
2365
select_lex->with_wild,
2366
select_lex->item_list,
2368
0, (ORDER *)NULL, (ORDER *)NULL, (Item *)NULL,
2370
select_lex->options | thd->options |
2371
SELECT_NO_JOIN_CACHE | SELECT_NO_UNLOCK |
2372
OPTION_SETUP_TABLES_DONE,
2373
del_result, unit, select_lex);
2374
res|= thd->is_error();
2376
del_result->abort();
2383
case SQLCOM_DROP_TABLE:
2385
DBUG_ASSERT(first_table == all_tables && first_table != 0);
2386
if (!lex->drop_temporary)
2388
if (end_active_trans(thd))
2394
If this is a slave thread, we may sometimes execute some
2395
DROP / * 40005 TEMPORARY * / TABLE
2396
that come from parts of binlogs (likely if we use RESET SLAVE or CHANGE
2397
MASTER TO), while the temporary table has already been dropped.
2398
To not generate such irrelevant "table does not exist errors",
2399
we silently add IF EXISTS if TEMPORARY was used.
2401
if (thd->slave_thread)
2402
lex->drop_if_exists= 1;
2404
/* So that DROP TEMPORARY TABLE gets to binlog at commit/rollback */
2405
thd->options|= OPTION_KEEP_LOG;
2407
/* DDL and binlog write order protected by LOCK_open */
2408
res= mysql_rm_table(thd, first_table, lex->drop_if_exists,
2409
lex->drop_temporary);
2412
case SQLCOM_SHOW_PROCESSLIST:
2413
mysqld_list_processes(thd, NullS, lex->verbose);
2415
case SQLCOM_SHOW_ENGINE_LOGS:
2417
res= ha_show_status(thd, lex->create_info.db_type, HA_ENGINE_LOGS);
2420
case SQLCOM_CHANGE_DB:
2422
LEX_STRING db_str= { (char *) select_lex->db, strlen(select_lex->db) };
2424
if (!mysql_change_db(thd, &db_str, FALSE))
2432
DBUG_ASSERT(first_table == all_tables && first_table != 0);
2433
if (lex->local_file)
2435
if (!(thd->client_capabilities & CLIENT_LOCAL_FILES) ||
2438
my_message(ER_NOT_ALLOWED_COMMAND, ER(ER_NOT_ALLOWED_COMMAND), MYF(0));
2443
res= mysql_load(thd, lex->exchange, first_table, lex->field_list,
2444
lex->update_list, lex->value_list, lex->duplicates,
2445
lex->ignore, (bool) lex->local_file);
2449
case SQLCOM_SET_OPTION:
2451
List<set_var_base> *lex_var_list= &lex->var_list;
2453
if (lex->autocommit && end_active_trans(thd))
2456
if (open_and_lock_tables(thd, all_tables))
2458
if (lex->one_shot_set && not_all_support_one_shot(lex_var_list))
2460
my_error(ER_RESERVED_SYNTAX, MYF(0), "SET ONE_SHOT");
2463
if (!(res= sql_set_variables(thd, lex_var_list)))
2466
If the previous command was a SET ONE_SHOT, we don't want to forget
2467
about the ONE_SHOT property of that SET. So we use a |= instead of = .
2469
thd->one_shot_set|= lex->one_shot_set;
2475
We encountered some sort of error, but no message was sent.
2476
Send something semi-generic here since we don't know which
2477
assignment in the list caused the error.
2479
if (!thd->is_error())
2480
my_error(ER_WRONG_ARGUMENTS,MYF(0),"SET");
2487
case SQLCOM_UNLOCK_TABLES:
2489
It is critical for mysqldump --single-transaction --master-data that
2490
UNLOCK TABLES does not implicitely commit a connection which has only
2491
done FLUSH TABLES WITH READ LOCK + BEGIN. If this assumption becomes
2492
false, mysqldump will not work.
2494
unlock_locked_tables(thd);
2495
if (thd->options & OPTION_TABLE_LOCK)
2497
end_active_trans(thd);
2498
thd->options&= ~(OPTION_TABLE_LOCK);
2500
if (thd->global_read_lock)
2501
unlock_global_read_lock(thd);
2504
case SQLCOM_LOCK_TABLES:
2506
We try to take transactional locks if
2507
- only transactional locks are requested (lex->lock_transactional) and
2508
- no non-transactional locks exist (!thd->locked_tables).
2510
DBUG_PRINT("lock_info", ("lex->lock_transactional: %d "
2511
"thd->locked_tables: 0x%lx",
2512
lex->lock_transactional,
2513
(long) thd->locked_tables));
2514
if (lex->lock_transactional && !thd->locked_tables)
2518
All requested locks are transactional and no non-transactional
2521
if ((rc= try_transactional_lock(thd, all_tables)) == -1)
2529
Non-transactional locking has been requested or
2530
non-transactional locks exist already or transactional locks are
2531
not supported by all storage engines. Take non-transactional
2536
One or more requested locks are non-transactional and/or
2537
non-transactional locks exist or a storage engine does not support
2538
transactional locks. Check if at least one transactional lock is
2539
requested. If yes, warn about the conversion to non-transactional
2540
locks or abort in strict mode.
2542
if (check_transactional_lock(thd, all_tables))
2544
unlock_locked_tables(thd);
2545
/* we must end the trasaction first, regardless of anything */
2546
if (end_active_trans(thd))
2548
thd->in_lock_tables=1;
2549
thd->options|= OPTION_TABLE_LOCK;
2551
if (!(res= simple_open_n_lock_tables(thd, all_tables)))
2553
thd->locked_tables=thd->lock;
2555
(void) set_handler_table_locks(thd, all_tables, FALSE);
2556
DBUG_PRINT("lock_info", ("thd->locked_tables: 0x%lx",
2557
(long) thd->locked_tables));
2563
Need to end the current transaction, so the storage engine (InnoDB)
2564
can free its locks if LOCK TABLES locked some tables before finding
2565
that it can't lock a table in its list
2567
ha_autocommit_or_rollback(thd, 1);
2568
end_active_trans(thd);
2569
thd->options&= ~(OPTION_TABLE_LOCK);
2571
thd->in_lock_tables=0;
2573
case SQLCOM_CREATE_DB:
2576
As mysql_create_db() may modify HA_CREATE_INFO structure passed to
2577
it, we need to use a copy of LEX::create_info to make execution
2578
prepared statement- safe.
2580
HA_CREATE_INFO create_info(lex->create_info);
2581
if (end_active_trans(thd))
2587
if (!(alias=thd->strmake(lex->name.str, lex->name.length)) ||
2588
check_db_name(&lex->name))
2590
my_error(ER_WRONG_DB_NAME, MYF(0), lex->name.str);
2594
If in a slave thread :
2595
CREATE DATABASE DB was certainly not preceded by USE DB.
2596
For that reason, db_ok() in sql/slave.cc did not check the
2597
do_db/ignore_db. And as this query involves no tables, tables_ok()
2598
above was not called. So we have to check rules again here.
2600
if (thd->slave_thread &&
2601
(!rpl_filter->db_ok(lex->name.str) ||
2602
!rpl_filter->db_ok_with_wild_table(lex->name.str)))
2604
my_message(ER_SLAVE_IGNORED_TABLE, ER(ER_SLAVE_IGNORED_TABLE), MYF(0));
2607
res= mysql_create_db(thd,(lower_case_table_names == 2 ? alias :
2608
lex->name.str), &create_info, 0);
2611
case SQLCOM_DROP_DB:
2613
if (end_active_trans(thd))
2618
if (check_db_name(&lex->name))
2620
my_error(ER_WRONG_DB_NAME, MYF(0), lex->name.str);
2624
If in a slave thread :
2625
DROP DATABASE DB may not be preceded by USE DB.
2626
For that reason, maybe db_ok() in sql/slave.cc did not check the
2627
do_db/ignore_db. And as this query involves no tables, tables_ok()
2628
above was not called. So we have to check rules again here.
2630
#ifdef HAVE_REPLICATION
2631
if (thd->slave_thread &&
2632
(!rpl_filter->db_ok(lex->name.str) ||
2633
!rpl_filter->db_ok_with_wild_table(lex->name.str)))
2635
my_message(ER_SLAVE_IGNORED_TABLE, ER(ER_SLAVE_IGNORED_TABLE), MYF(0));
2639
if (thd->locked_tables || thd->active_transaction())
2641
my_message(ER_LOCK_OR_ACTIVE_TRANSACTION,
2642
ER(ER_LOCK_OR_ACTIVE_TRANSACTION), MYF(0));
2645
res= mysql_rm_db(thd, lex->name.str, lex->drop_if_exists, 0);
2648
case SQLCOM_ALTER_DB_UPGRADE:
2650
LEX_STRING *db= & lex->name;
2651
if (end_active_trans(thd))
2656
#ifdef HAVE_REPLICATION
2657
if (thd->slave_thread &&
2658
(!rpl_filter->db_ok(db->str) ||
2659
!rpl_filter->db_ok_with_wild_table(db->str)))
2662
my_message(ER_SLAVE_IGNORED_TABLE, ER(ER_SLAVE_IGNORED_TABLE), MYF(0));
2666
if (check_db_name(db))
2668
my_error(ER_WRONG_DB_NAME, MYF(0), db->str);
2671
if (thd->locked_tables || thd->active_transaction())
2674
my_message(ER_LOCK_OR_ACTIVE_TRANSACTION,
2675
ER(ER_LOCK_OR_ACTIVE_TRANSACTION), MYF(0));
2679
res= mysql_upgrade_db(thd, db);
2684
case SQLCOM_ALTER_DB:
2686
LEX_STRING *db= &lex->name;
2687
HA_CREATE_INFO create_info(lex->create_info);
2688
if (check_db_name(db))
2690
my_error(ER_WRONG_DB_NAME, MYF(0), db->str);
2694
If in a slave thread :
2695
ALTER DATABASE DB may not be preceded by USE DB.
2696
For that reason, maybe db_ok() in sql/slave.cc did not check the
2697
do_db/ignore_db. And as this query involves no tables, tables_ok()
2698
above was not called. So we have to check rules again here.
2700
#ifdef HAVE_REPLICATION
2701
if (thd->slave_thread &&
2702
(!rpl_filter->db_ok(db->str) ||
2703
!rpl_filter->db_ok_with_wild_table(db->str)))
2705
my_message(ER_SLAVE_IGNORED_TABLE, ER(ER_SLAVE_IGNORED_TABLE), MYF(0));
2709
if (thd->locked_tables || thd->active_transaction())
2711
my_message(ER_LOCK_OR_ACTIVE_TRANSACTION,
2712
ER(ER_LOCK_OR_ACTIVE_TRANSACTION), MYF(0));
2715
res= mysql_alter_db(thd, db->str, &create_info);
2718
case SQLCOM_SHOW_CREATE_DB:
2720
DBUG_EXECUTE_IF("4x_server_emul",
2721
my_error(ER_UNKNOWN_ERROR, MYF(0)); goto error;);
2722
if (check_db_name(&lex->name))
2724
my_error(ER_WRONG_DB_NAME, MYF(0), lex->name.str);
2727
res= mysqld_show_create_db(thd, lex->name.str, &lex->create_info);
2732
RESET commands are never written to the binary log, so we have to
2733
initialize this variable because RESET shares the same code as FLUSH
2735
lex->no_write_to_binlog= 1;
2738
bool write_to_binlog;
2741
reload_cache() will tell us if we are allowed to write to the
2744
if (!reload_cache(thd, lex->type, first_table, &write_to_binlog))
2747
We WANT to write and we CAN write.
2748
! we write after unlocking the table.
2751
Presumably, RESET and binlog writing doesn't require synchronization
2753
if (!lex->no_write_to_binlog && write_to_binlog)
2755
write_bin_log(thd, FALSE, thd->query, thd->query_length);
2764
Item *it= (Item *)lex->value_list.head();
2766
if (lex->table_or_sp_used())
2768
my_error(ER_NOT_SUPPORTED_YET, MYF(0), "Usage of subqueries or stored "
2769
"function calls as part of this statement");
2773
if ((!it->fixed && it->fix_fields(lex->thd, &it)) || it->check_cols(1))
2775
my_message(ER_SET_CONSTANTS_ONLY, ER(ER_SET_CONSTANTS_ONLY),
2779
sql_kill(thd, (ulong)it->val_int(), lex->type & ONLY_KILL_QUERY);
2783
if (thd->transaction.xid_state.xa_state != XA_NOTR)
2785
my_error(ER_XAER_RMFAIL, MYF(0),
2786
xa_state_names[thd->transaction.xid_state.xa_state]);
2790
Breakpoints for backup testing.
2792
if (begin_trans(thd))
2797
if (end_trans(thd, lex->tx_release ? COMMIT_RELEASE :
2798
lex->tx_chain ? COMMIT_AND_CHAIN : COMMIT))
2802
case SQLCOM_ROLLBACK:
2803
if (end_trans(thd, lex->tx_release ? ROLLBACK_RELEASE :
2804
lex->tx_chain ? ROLLBACK_AND_CHAIN : ROLLBACK))
2808
case SQLCOM_RELEASE_SAVEPOINT:
2811
for (sv=thd->transaction.savepoints; sv; sv=sv->prev)
2813
if (my_strnncoll(system_charset_info,
2814
(uchar *)lex->ident.str, lex->ident.length,
2815
(uchar *)sv->name, sv->length) == 0)
2820
if (ha_release_savepoint(thd, sv))
2821
res= TRUE; // cannot happen
2824
thd->transaction.savepoints=sv->prev;
2827
my_error(ER_SP_DOES_NOT_EXIST, MYF(0), "SAVEPOINT", lex->ident.str);
2830
case SQLCOM_ROLLBACK_TO_SAVEPOINT:
2833
for (sv=thd->transaction.savepoints; sv; sv=sv->prev)
2835
if (my_strnncoll(system_charset_info,
2836
(uchar *)lex->ident.str, lex->ident.length,
2837
(uchar *)sv->name, sv->length) == 0)
2842
if (ha_rollback_to_savepoint(thd, sv))
2843
res= TRUE; // cannot happen
2846
if (((thd->options & OPTION_KEEP_LOG) ||
2847
thd->transaction.all.modified_non_trans_table) &&
2849
push_warning(thd, MYSQL_ERROR::WARN_LEVEL_WARN,
2850
ER_WARNING_NOT_COMPLETE_ROLLBACK,
2851
ER(ER_WARNING_NOT_COMPLETE_ROLLBACK));
2854
thd->transaction.savepoints=sv;
2857
my_error(ER_SP_DOES_NOT_EXIST, MYF(0), "SAVEPOINT", lex->ident.str);
2860
case SQLCOM_SAVEPOINT:
2861
if (!(thd->options & (OPTION_NOT_AUTOCOMMIT | OPTION_BEGIN) ||
2862
thd->in_sub_stmt) || !opt_using_transactions)
2866
SAVEPOINT **sv, *newsv;
2867
for (sv=&thd->transaction.savepoints; *sv; sv=&(*sv)->prev)
2869
if (my_strnncoll(system_charset_info,
2870
(uchar *)lex->ident.str, lex->ident.length,
2871
(uchar *)(*sv)->name, (*sv)->length) == 0)
2874
if (*sv) /* old savepoint of the same name exists */
2877
ha_release_savepoint(thd, *sv); // it cannot fail
2880
else if ((newsv=(SAVEPOINT *) alloc_root(&thd->transaction.mem_root,
2881
savepoint_alloc_size)) == 0)
2883
my_error(ER_OUT_OF_RESOURCES, MYF(0));
2886
newsv->name=strmake_root(&thd->transaction.mem_root,
2887
lex->ident.str, lex->ident.length);
2888
newsv->length=lex->ident.length;
2890
if we'll get an error here, don't add new savepoint to the list.
2891
we'll lose a little bit of memory in transaction mem_root, but it'll
2892
be free'd when transaction ends anyway
2894
if (ha_savepoint(thd, newsv))
2898
newsv->prev=thd->transaction.savepoints;
2899
thd->transaction.savepoints=newsv;
2904
case SQLCOM_BINLOG_BASE64_EVENT:
2906
mysql_client_binlog_statement(thd);
2910
DBUG_ASSERT(0); /* Impossible */
2914
thd_proc_info(thd, "query end");
2917
Binlog-related cleanup:
2918
Reset system variables temporarily modified by SET ONE SHOT.
2920
Exception: If this is a SET, do nothing. This is to allow
2921
mysqlbinlog to print many SET commands (in this case we want the
2922
charset temp setting to live until the real query). This is also
2923
needed so that SET CHARACTER_SET_CLIENT... does not cancel itself
2926
if (thd->one_shot_set && lex->sql_command != SQLCOM_SET_OPTION)
2927
reset_one_shot_variables(thd);
514
2930
The return value for ROW_COUNT() is "implementation dependent" if the