Server IP : 172.67.216.182 / Your IP : 162.158.189.100 Web Server : Apache System : Linux krdc-ubuntu-s-2vcpu-4gb-amd-blr1-01.localdomain 5.15.0-142-generic #152-Ubuntu SMP Mon May 19 10:54:31 UTC 2025 x86_64 User : www ( 1000) PHP Version : 7.4.33 Disable Function : passthru,exec,system,putenv,chroot,chgrp,chown,shell_exec,popen,proc_open,pcntl_exec,ini_alter,ini_restore,dl,openlog,syslog,readlink,symlink,popepassthru,pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,imap_open,apache_setenv MySQL : OFF | cURL : ON | WGET : ON | Perl : ON | Python : OFF | Sudo : ON | Pkexec : ON Directory : /www/server/mysql/src/sql/ |
Upload File : |
/* Copyright (c) 2000, 2023, Oracle and/or its affiliates. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License, version 2.0, as published by the Free Software Foundation. This program is also distributed with certain software (including but not limited to OpenSSL) that is licensed under separate terms, as designated in a particular file or component or in included license documentation. The authors of MySQL hereby grant you an additional permission to link the program and your derivative works with the separately licensed software that they have included with MySQL. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License, version 2.0, for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #include "sql_parse.h" #include "auth_common.h" // acl_authenticate #include "binlog.h" // purge_master_logs #include "debug_sync.h" // DEBUG_SYNC #include "events.h" // Events #include "item_timefunc.h" // Item_func_unix_timestamp #include "log.h" // query_logger #include "log_event.h" // slave_execute_deferred_events #include "mysys_err.h" // EE_CAPACITY_EXCEEDED #include "opt_explain.h" // mysql_explain_other #include "opt_trace.h" // Opt_trace_start #include "partition_info.h" // partition_info #include "probes_mysql.h" // MYSQL_COMMAND_START #include "rpl_filter.h" // rpl_filter #include "rpl_master.h" // register_slave #include "rpl_rli.h" // mysql_show_relaylog_events #include "rpl_slave.h" // change_master_cmd #include "sp.h" // sp_create_routine #include "sp_cache.h" // sp_cache_enforce_limit #include "sp_head.h" // sp_head #include "sql_admin.h" // mysql_assign_to_keycache #include "sql_analyse.h" // Query_result_analyse #include "sql_audit.h" // MYSQL_AUDIT_NOTIFY_CONNECTION_CHANGE_USER #include "sql_base.h" // find_temporary_table #include "sql_binlog.h" // mysql_client_binlog_statement #include "sql_cache.h" // query_cache #include "sql_connect.h" // decrease_user_connections #include "sql_db.h" // mysql_change_db #include "sql_delete.h" // mysql_delete #include "sql_do.h" // mysql_do #include "sql_handler.h" // mysql_ha_rm_tables #include "sql_help.h" // mysqld_help #include "sql_insert.h" // Query_result_create #include "sql_load.h" // mysql_load #include "sql_prepare.h" // mysql_stmt_execute #include "partition_info.h" // has_external_data_or_index_dir #include "sql_reload.h" // reload_acl_and_cache #include "sql_rename.h" // mysql_rename_tables #include "sql_select.h" // handle_query #include "sql_show.h" // find_schema_table #include "sql_table.h" // mysql_create_table #include "sql_tablespace.h" // mysql_alter_tablespace #include "sql_test.h" // mysql_print_status #include "sql_select.h" // handle_query #include "sql_load.h" // mysql_load #include "sql_servers.h" // create_servers, alter_servers, // drop_servers, servers_reload #include "sql_handler.h" // mysql_ha_open, mysql_ha_close, // mysql_ha_read #include "sql_binlog.h" // mysql_client_binlog_statement #include "sql_do.h" // mysql_do #include "sql_help.h" // mysqld_help #include "rpl_constants.h" // Incident, INCIDENT_LOST_EVENTS #include "log_event.h" #include "rpl_slave.h" #include "rpl_master.h" #include "rpl_msr.h" /* Multisource replication */ #include "rpl_filter.h" #include <m_ctype.h> #include <myisam.h> #include <my_dir.h> #include <dur_prop.h> #include "rpl_handler.h" #include "sp_head.h" #include "sp.h" #include "sp_cache.h" #include "events.h" #include "sql_trigger.h" // mysql_create_or_drop_trigger #include "transaction.h" #include "xa.h" #include "sql_audit.h" #include "sql_prepare.h" #include "debug_sync.h" #include "probes_mysql.h" #include "set_var.h" #include "opt_trace.h" #include "mysql/psi/mysql_statement.h" #include "opt_explain.h" #include "sql_rewrite.h" #include "sql_analyse.h" #include "table_cache.h" // table_cache_manager #include "sql_timer.h" // thd_timer_set, thd_timer_reset #include "sp_rcontext.h" #include "parse_location.h" #include "sql_digest.h" #include "sql_timer.h" // thd_timer_set #include "sql_trigger.h" // add_table_for_trigger #include "sql_update.h" // mysql_update #include "sql_view.h" // mysql_create_view #include "table_cache.h" // table_cache_manager #include "transaction.h" // trans_commit_implicit #include "sql_query_rewrite.h" #include "rpl_group_replication.h" #include <algorithm> using std::max; /** @defgroup Runtime_Environment Runtime Environment @{ */ /* Used in error handling only */ #define SP_COM_STRING(LP) \ ((LP)->sql_command == SQLCOM_CREATE_SPFUNCTION || \ (LP)->sql_command == SQLCOM_ALTER_FUNCTION || \ (LP)->sql_command == SQLCOM_SHOW_CREATE_FUNC || \ (LP)->sql_command == SQLCOM_DROP_FUNCTION ? \ "FUNCTION" : "PROCEDURE") static void sql_kill(THD *thd, my_thread_id id, bool only_kill_query); const LEX_STRING command_name[]={ { C_STRING_WITH_LEN("Sleep") }, { C_STRING_WITH_LEN("Quit") }, { C_STRING_WITH_LEN("Init DB") }, { C_STRING_WITH_LEN("Query") }, { C_STRING_WITH_LEN("Field List") }, { C_STRING_WITH_LEN("Create DB") }, { C_STRING_WITH_LEN("Drop DB") }, { C_STRING_WITH_LEN("Refresh") }, { C_STRING_WITH_LEN("Shutdown") }, { C_STRING_WITH_LEN("Statistics") }, { C_STRING_WITH_LEN("Processlist") }, { C_STRING_WITH_LEN("Connect") }, { C_STRING_WITH_LEN("Kill") }, { C_STRING_WITH_LEN("Debug") }, { C_STRING_WITH_LEN("Ping") }, { C_STRING_WITH_LEN("Time") }, { C_STRING_WITH_LEN("Delayed insert") }, { C_STRING_WITH_LEN("Change user") }, { C_STRING_WITH_LEN("Binlog Dump") }, { C_STRING_WITH_LEN("Table Dump") }, { C_STRING_WITH_LEN("Connect Out") }, { C_STRING_WITH_LEN("Register Slave") }, { C_STRING_WITH_LEN("Prepare") }, { C_STRING_WITH_LEN("Execute") }, { C_STRING_WITH_LEN("Long Data") }, { C_STRING_WITH_LEN("Close stmt") }, { C_STRING_WITH_LEN("Reset stmt") }, { C_STRING_WITH_LEN("Set option") }, { C_STRING_WITH_LEN("Fetch") }, { C_STRING_WITH_LEN("Daemon") }, { C_STRING_WITH_LEN("Binlog Dump GTID") }, { C_STRING_WITH_LEN("Reset Connection") }, { C_STRING_WITH_LEN("Error") } // Last command number }; #ifdef HAVE_REPLICATION /** Returns true if all tables should be ignored. */ bool all_tables_not_ok(THD *thd, TABLE_LIST *tables) { return rpl_filter->is_on() && tables && !thd->sp_runtime_ctx && !rpl_filter->tables_ok(thd->db().str, tables); } /** Checks whether the event for the given database, db, should be ignored or not. This is done by checking whether there are active rules in ignore_db or in do_db containers. If there are, then check if there is a match, if not then check the wild_do rules. NOTE: This means that when using this function replicate-do-db and replicate-ignore-db take precedence over wild do rules. @param thd Thread handle. @param db Database name used while evaluating the filtering rules. @param sql_cmd Represents the current query that needs to be verified against the database filter rules. @return TRUE Query should not be filtered out from the execution. FALSE Query should be filtered out from the execution. */ inline bool check_database_filters(THD *thd, const char* db, enum_sql_command sql_cmd) { DBUG_ENTER("check_database_filters"); assert(thd->slave_thread); if (!db) DBUG_RETURN(TRUE); switch (sql_cmd) { case SQLCOM_BEGIN: case SQLCOM_COMMIT: case SQLCOM_SAVEPOINT: case SQLCOM_ROLLBACK: case SQLCOM_ROLLBACK_TO_SAVEPOINT: DBUG_RETURN(TRUE); default: break; } bool db_ok= rpl_filter->db_ok(db); /* No filters exist in ignore/do_db ? Then, just check wild_do_table filtering for 'DATABASE' related statements (CREATE/DROP/ATLER DATABASE) */ if (db_ok && (rpl_filter->get_do_db()->is_empty() && rpl_filter->get_ignore_db()->is_empty())) { switch (sql_cmd) { case SQLCOM_CREATE_DB: case SQLCOM_ALTER_DB: case SQLCOM_ALTER_DB_UPGRADE: case SQLCOM_DROP_DB: db_ok= rpl_filter->db_ok_with_wild_table(db); default: break; } } DBUG_RETURN(db_ok); } #endif bool some_non_temp_table_to_be_updated(THD *thd, TABLE_LIST *tables) { for (TABLE_LIST *table= tables; table; table= table->next_global) { assert(table->db && table->table_name); /* Update on performance_schema and temp tables are allowed in readonly mode. */ if (table->updating && !find_temporary_table(thd, table) && !is_perfschema_db(table->db, table->db_length)) return 1; } return 0; } /* Implicitly commit a active transaction if statement requires so. @param thd Thread handle. @param mask Bitmask used for the SQL command match. */ bool stmt_causes_implicit_commit(const THD *thd, uint mask) { const LEX *lex= thd->lex; bool skip= FALSE; DBUG_ENTER("stmt_causes_implicit_commit"); if (!(sql_command_flags[lex->sql_command] & mask)) DBUG_RETURN(FALSE); switch (lex->sql_command) { case SQLCOM_DROP_TABLE: skip= lex->drop_temporary; break; case SQLCOM_ALTER_TABLE: case SQLCOM_CREATE_TABLE: /* If CREATE TABLE of non-temporary table, do implicit commit */ skip= (lex->create_info.options & HA_LEX_CREATE_TMP_TABLE); break; case SQLCOM_SET_OPTION: /* Implicitly commit a transaction started by a SET statement */ skip= lex->autocommit ? FALSE : TRUE; break; default: break; } DBUG_RETURN(!skip); } /** Mark all commands that somehow changes a table. This is used to check number of updates / hour. sql_command is actually set to SQLCOM_END sometimes so we need the +1 to include it in the array. See COMMAND_FLAG_xxx for different type of commands 2 - query that returns meaningful ROW_COUNT() - a number of modified rows */ uint sql_command_flags[SQLCOM_END+1]; uint server_command_flags[COM_END+1]; void init_update_queries(void) { /* Initialize the server command flags array. */ memset(server_command_flags, 0, sizeof(server_command_flags)); server_command_flags[COM_SLEEP]= CF_ALLOW_PROTOCOL_PLUGIN; server_command_flags[COM_INIT_DB]= CF_ALLOW_PROTOCOL_PLUGIN; server_command_flags[COM_QUERY]= CF_ALLOW_PROTOCOL_PLUGIN; server_command_flags[COM_FIELD_LIST]= CF_ALLOW_PROTOCOL_PLUGIN; server_command_flags[COM_REFRESH]= CF_ALLOW_PROTOCOL_PLUGIN; server_command_flags[COM_SHUTDOWN]= CF_ALLOW_PROTOCOL_PLUGIN; server_command_flags[COM_STATISTICS]= CF_SKIP_QUESTIONS; server_command_flags[COM_PROCESS_KILL]= CF_ALLOW_PROTOCOL_PLUGIN; server_command_flags[COM_PING]= CF_SKIP_QUESTIONS; server_command_flags[COM_STMT_PREPARE]= CF_SKIP_QUESTIONS | CF_ALLOW_PROTOCOL_PLUGIN; server_command_flags[COM_STMT_EXECUTE]= CF_ALLOW_PROTOCOL_PLUGIN; server_command_flags[COM_STMT_SEND_LONG_DATA]= CF_ALLOW_PROTOCOL_PLUGIN; server_command_flags[COM_STMT_CLOSE]= CF_SKIP_QUESTIONS | CF_ALLOW_PROTOCOL_PLUGIN; server_command_flags[COM_STMT_RESET]= CF_SKIP_QUESTIONS | CF_ALLOW_PROTOCOL_PLUGIN; server_command_flags[COM_STMT_FETCH]= CF_ALLOW_PROTOCOL_PLUGIN; server_command_flags[COM_END]= CF_ALLOW_PROTOCOL_PLUGIN; /* Initialize the sql command flags array. */ memset(sql_command_flags, 0, sizeof(sql_command_flags)); /* In general, DDL statements do not generate row events and do not go through a cache before being written to the binary log. However, the CREATE TABLE...SELECT is an exception because it may generate row events. For that reason, the SQLCOM_CREATE_TABLE which represents a CREATE TABLE, including the CREATE TABLE...SELECT, has the CF_CAN_GENERATE_ROW_EVENTS flag. The distinction between a regular CREATE TABLE and the CREATE TABLE...SELECT is made in other parts of the code, in particular in the Query_log_event's constructor. */ sql_command_flags[SQLCOM_CREATE_TABLE]= CF_CHANGES_DATA | CF_REEXECUTION_FRAGILE | CF_AUTO_COMMIT_TRANS | CF_CAN_GENERATE_ROW_EVENTS; sql_command_flags[SQLCOM_CREATE_INDEX]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_ALTER_TABLE]= CF_CHANGES_DATA | CF_WRITE_LOGS_COMMAND | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_TRUNCATE]= CF_CHANGES_DATA | CF_WRITE_LOGS_COMMAND | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_DROP_TABLE]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_LOAD]= CF_CHANGES_DATA | CF_REEXECUTION_FRAGILE | CF_CAN_GENERATE_ROW_EVENTS; sql_command_flags[SQLCOM_CREATE_DB]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_DROP_DB]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_ALTER_DB_UPGRADE]= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_ALTER_DB]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_RENAME_TABLE]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_DROP_INDEX]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_CREATE_VIEW]= CF_CHANGES_DATA | CF_REEXECUTION_FRAGILE | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_DROP_VIEW]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_CREATE_TRIGGER]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_DROP_TRIGGER]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_CREATE_EVENT]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_ALTER_EVENT]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_DROP_EVENT]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_UPDATE]= CF_CHANGES_DATA | CF_REEXECUTION_FRAGILE | CF_CAN_GENERATE_ROW_EVENTS | CF_OPTIMIZER_TRACE | CF_CAN_BE_EXPLAINED; sql_command_flags[SQLCOM_UPDATE_MULTI]= CF_CHANGES_DATA | CF_REEXECUTION_FRAGILE | CF_CAN_GENERATE_ROW_EVENTS | CF_OPTIMIZER_TRACE | CF_CAN_BE_EXPLAINED; // This is INSERT VALUES(...), can be VALUES(stored_func()) so we trace it sql_command_flags[SQLCOM_INSERT]= CF_CHANGES_DATA | CF_REEXECUTION_FRAGILE | CF_CAN_GENERATE_ROW_EVENTS | CF_OPTIMIZER_TRACE | CF_CAN_BE_EXPLAINED; sql_command_flags[SQLCOM_INSERT_SELECT]= CF_CHANGES_DATA | CF_REEXECUTION_FRAGILE | CF_CAN_GENERATE_ROW_EVENTS | CF_OPTIMIZER_TRACE | CF_CAN_BE_EXPLAINED; sql_command_flags[SQLCOM_DELETE]= CF_CHANGES_DATA | CF_REEXECUTION_FRAGILE | CF_CAN_GENERATE_ROW_EVENTS | CF_OPTIMIZER_TRACE | CF_CAN_BE_EXPLAINED; sql_command_flags[SQLCOM_DELETE_MULTI]= CF_CHANGES_DATA | CF_REEXECUTION_FRAGILE | CF_CAN_GENERATE_ROW_EVENTS | CF_OPTIMIZER_TRACE | CF_CAN_BE_EXPLAINED; sql_command_flags[SQLCOM_REPLACE]= CF_CHANGES_DATA | CF_REEXECUTION_FRAGILE | CF_CAN_GENERATE_ROW_EVENTS | CF_OPTIMIZER_TRACE | CF_CAN_BE_EXPLAINED; sql_command_flags[SQLCOM_REPLACE_SELECT]= CF_CHANGES_DATA | CF_REEXECUTION_FRAGILE | CF_CAN_GENERATE_ROW_EVENTS | CF_OPTIMIZER_TRACE | CF_CAN_BE_EXPLAINED; sql_command_flags[SQLCOM_SELECT]= CF_REEXECUTION_FRAGILE | CF_CAN_GENERATE_ROW_EVENTS | CF_OPTIMIZER_TRACE | CF_CAN_BE_EXPLAINED; // (1) so that subquery is traced when doing "SET @var = (subquery)" /* @todo SQLCOM_SET_OPTION should have CF_CAN_GENERATE_ROW_EVENTS set, because it may invoke a stored function that generates row events. /Sven */ sql_command_flags[SQLCOM_SET_OPTION]= CF_REEXECUTION_FRAGILE | CF_AUTO_COMMIT_TRANS | CF_CAN_GENERATE_ROW_EVENTS | CF_OPTIMIZER_TRACE; // (1) // (1) so that subquery is traced when doing "DO @var := (subquery)" sql_command_flags[SQLCOM_DO]= CF_REEXECUTION_FRAGILE | CF_CAN_GENERATE_ROW_EVENTS | CF_OPTIMIZER_TRACE; // (1) sql_command_flags[SQLCOM_SHOW_STATUS_PROC]= CF_STATUS_COMMAND | CF_REEXECUTION_FRAGILE; sql_command_flags[SQLCOM_SHOW_STATUS]= CF_STATUS_COMMAND | CF_REEXECUTION_FRAGILE; sql_command_flags[SQLCOM_SHOW_DATABASES]= CF_STATUS_COMMAND | CF_REEXECUTION_FRAGILE; sql_command_flags[SQLCOM_SHOW_TRIGGERS]= CF_STATUS_COMMAND | CF_REEXECUTION_FRAGILE; sql_command_flags[SQLCOM_SHOW_EVENTS]= CF_STATUS_COMMAND | CF_REEXECUTION_FRAGILE; sql_command_flags[SQLCOM_SHOW_OPEN_TABLES]= CF_STATUS_COMMAND | CF_REEXECUTION_FRAGILE; sql_command_flags[SQLCOM_SHOW_PLUGINS]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_FIELDS]= CF_STATUS_COMMAND | CF_REEXECUTION_FRAGILE; sql_command_flags[SQLCOM_SHOW_KEYS]= CF_STATUS_COMMAND | CF_REEXECUTION_FRAGILE; sql_command_flags[SQLCOM_SHOW_VARIABLES]= CF_STATUS_COMMAND | CF_REEXECUTION_FRAGILE; sql_command_flags[SQLCOM_SHOW_CHARSETS]= CF_STATUS_COMMAND | CF_REEXECUTION_FRAGILE; sql_command_flags[SQLCOM_SHOW_COLLATIONS]= CF_STATUS_COMMAND | CF_REEXECUTION_FRAGILE; sql_command_flags[SQLCOM_SHOW_BINLOGS]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_SLAVE_HOSTS]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_BINLOG_EVENTS]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_STORAGE_ENGINES]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_PRIVILEGES]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_WARNS]= CF_STATUS_COMMAND | CF_DIAGNOSTIC_STMT; sql_command_flags[SQLCOM_SHOW_ERRORS]= CF_STATUS_COMMAND | CF_DIAGNOSTIC_STMT; sql_command_flags[SQLCOM_SHOW_ENGINE_STATUS]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_ENGINE_MUTEX]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_ENGINE_LOGS]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_PROCESSLIST]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_GRANTS]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_CREATE_DB]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_CREATE]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_MASTER_STAT]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_SLAVE_STAT]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_CREATE_PROC]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_CREATE_FUNC]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_CREATE_TRIGGER]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_STATUS_FUNC]= CF_STATUS_COMMAND | CF_REEXECUTION_FRAGILE; sql_command_flags[SQLCOM_SHOW_PROC_CODE]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_FUNC_CODE]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_CREATE_EVENT]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_PROFILES]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_SHOW_PROFILE]= CF_STATUS_COMMAND; sql_command_flags[SQLCOM_BINLOG_BASE64_EVENT]= CF_STATUS_COMMAND | CF_CAN_GENERATE_ROW_EVENTS; sql_command_flags[SQLCOM_SHOW_TABLES]= (CF_STATUS_COMMAND | CF_SHOW_TABLE_COMMAND | CF_REEXECUTION_FRAGILE); sql_command_flags[SQLCOM_SHOW_TABLE_STATUS]= (CF_STATUS_COMMAND | CF_SHOW_TABLE_COMMAND | CF_REEXECUTION_FRAGILE); sql_command_flags[SQLCOM_CREATE_USER]= CF_CHANGES_DATA; sql_command_flags[SQLCOM_RENAME_USER]= CF_CHANGES_DATA; sql_command_flags[SQLCOM_DROP_USER]= CF_CHANGES_DATA; sql_command_flags[SQLCOM_ALTER_USER]= CF_CHANGES_DATA; sql_command_flags[SQLCOM_GRANT]= CF_CHANGES_DATA; sql_command_flags[SQLCOM_REVOKE]= CF_CHANGES_DATA; sql_command_flags[SQLCOM_REVOKE_ALL]= CF_CHANGES_DATA; sql_command_flags[SQLCOM_OPTIMIZE]= CF_CHANGES_DATA; sql_command_flags[SQLCOM_ALTER_INSTANCE]= CF_CHANGES_DATA; sql_command_flags[SQLCOM_CREATE_FUNCTION]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_CREATE_PROCEDURE]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_CREATE_SPFUNCTION]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_DROP_PROCEDURE]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_DROP_FUNCTION]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_ALTER_PROCEDURE]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_ALTER_FUNCTION]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_INSTALL_PLUGIN]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_UNINSTALL_PLUGIN]= CF_CHANGES_DATA | CF_AUTO_COMMIT_TRANS; /* Does not change the contents of the Diagnostics Area. */ sql_command_flags[SQLCOM_GET_DIAGNOSTICS]= CF_DIAGNOSTIC_STMT; /* (1): without it, in "CALL some_proc((subq))", subquery would not be traced. */ sql_command_flags[SQLCOM_CALL]= CF_REEXECUTION_FRAGILE | CF_CAN_GENERATE_ROW_EVENTS | CF_OPTIMIZER_TRACE; // (1) sql_command_flags[SQLCOM_EXECUTE]= CF_CAN_GENERATE_ROW_EVENTS; /* The following admin table operations are allowed on log tables. */ sql_command_flags[SQLCOM_REPAIR]= CF_WRITE_LOGS_COMMAND | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_OPTIMIZE]|= CF_WRITE_LOGS_COMMAND | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_ANALYZE]= CF_WRITE_LOGS_COMMAND | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_CHECK]= CF_WRITE_LOGS_COMMAND | CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_CREATE_USER]|= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_DROP_USER]|= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_RENAME_USER]|= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_ALTER_USER]|= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_REVOKE]|= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_REVOKE_ALL]|= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_GRANT]|= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_ASSIGN_TO_KEYCACHE]= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_PRELOAD_KEYS]= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_ALTER_INSTANCE]|= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_FLUSH]= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_RESET]= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_CREATE_SERVER]= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_ALTER_SERVER]= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_DROP_SERVER]= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_CHANGE_MASTER]= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_CHANGE_REPLICATION_FILTER]= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_SLAVE_START]= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_SLAVE_STOP]= CF_AUTO_COMMIT_TRANS; sql_command_flags[SQLCOM_ALTER_TABLESPACE]|= CF_AUTO_COMMIT_TRANS; /* The following statements can deal with temporary tables, so temporary tables should be pre-opened for those statements to simplify privilege checking. There are other statements that deal with temporary tables and open them, but which are not listed here. The thing is that the order of pre-opening temporary tables for those statements is somewhat custom. */ sql_command_flags[SQLCOM_CREATE_TABLE]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_DROP_TABLE]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_CREATE_INDEX]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_ALTER_TABLE]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_TRUNCATE]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_LOAD]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_DROP_INDEX]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_UPDATE]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_UPDATE_MULTI]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_INSERT_SELECT]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_DELETE]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_DELETE_MULTI]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_REPLACE_SELECT]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_SELECT]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_SET_OPTION]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_DO]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_CALL]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_CHECKSUM]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_ANALYZE]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_CHECK]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_OPTIMIZE]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_REPAIR]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_PRELOAD_KEYS]|= CF_PREOPEN_TMP_TABLES; sql_command_flags[SQLCOM_ASSIGN_TO_KEYCACHE]|= CF_PREOPEN_TMP_TABLES; /* DDL statements that should start with closing opened handlers. We use this flag only for statements for which open HANDLERs have to be closed before emporary tables are pre-opened. */ sql_command_flags[SQLCOM_CREATE_TABLE]|= CF_HA_CLOSE; sql_command_flags[SQLCOM_DROP_TABLE]|= CF_HA_CLOSE; sql_command_flags[SQLCOM_ALTER_TABLE]|= CF_HA_CLOSE; sql_command_flags[SQLCOM_TRUNCATE]|= CF_HA_CLOSE; sql_command_flags[SQLCOM_REPAIR]|= CF_HA_CLOSE; sql_command_flags[SQLCOM_OPTIMIZE]|= CF_HA_CLOSE; sql_command_flags[SQLCOM_ANALYZE]|= CF_HA_CLOSE; sql_command_flags[SQLCOM_CHECK]|= CF_HA_CLOSE; sql_command_flags[SQLCOM_CREATE_INDEX]|= CF_HA_CLOSE; sql_command_flags[SQLCOM_DROP_INDEX]|= CF_HA_CLOSE; sql_command_flags[SQLCOM_PRELOAD_KEYS]|= CF_HA_CLOSE; sql_command_flags[SQLCOM_ASSIGN_TO_KEYCACHE]|= CF_HA_CLOSE; /* Mark statements that always are disallowed in read-only transactions. Note that according to the SQL standard, even temporary table DDL should be disallowed. */ sql_command_flags[SQLCOM_CREATE_TABLE]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_ALTER_TABLE]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_DROP_TABLE]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_RENAME_TABLE]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_CREATE_INDEX]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_DROP_INDEX]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_CREATE_DB]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_DROP_DB]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_ALTER_DB_UPGRADE]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_ALTER_DB]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_CREATE_VIEW]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_DROP_VIEW]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_CREATE_TRIGGER]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_DROP_TRIGGER]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_CREATE_EVENT]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_ALTER_EVENT]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_DROP_EVENT]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_CREATE_USER]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_RENAME_USER]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_ALTER_USER]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_DROP_USER]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_CREATE_SERVER]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_ALTER_SERVER]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_DROP_SERVER]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_CREATE_FUNCTION]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_CREATE_PROCEDURE]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_CREATE_SPFUNCTION]|=CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_DROP_PROCEDURE]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_DROP_FUNCTION]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_ALTER_PROCEDURE]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_ALTER_FUNCTION]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_TRUNCATE]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_ALTER_TABLESPACE]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_REPAIR]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_OPTIMIZE]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_GRANT]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_REVOKE]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_REVOKE_ALL]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_INSTALL_PLUGIN]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_UNINSTALL_PLUGIN]|= CF_DISALLOW_IN_RO_TRANS; sql_command_flags[SQLCOM_ALTER_INSTANCE]|= CF_DISALLOW_IN_RO_TRANS; /* Mark statements that are allowed to be executed by the plugins. */ sql_command_flags[SQLCOM_SELECT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CREATE_TABLE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CREATE_INDEX]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_ALTER_TABLE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_UPDATE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_INSERT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_INSERT_SELECT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_DELETE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_TRUNCATE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_DROP_TABLE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_DROP_INDEX]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_DATABASES]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_TABLES]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_FIELDS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_KEYS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_VARIABLES]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_STATUS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_ENGINE_LOGS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_ENGINE_STATUS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_ENGINE_MUTEX]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_PROCESSLIST]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_MASTER_STAT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_SLAVE_STAT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_GRANTS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_CREATE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_CHARSETS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_COLLATIONS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_CREATE_DB]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_TABLE_STATUS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_TRIGGERS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_LOAD]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SET_OPTION]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_LOCK_TABLES]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_UNLOCK_TABLES]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_GRANT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CHANGE_DB]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CREATE_DB]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_DROP_DB]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_ALTER_DB]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_REPAIR]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_REPLACE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_REPLACE_SELECT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CREATE_FUNCTION]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_DROP_FUNCTION]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_REVOKE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_OPTIMIZE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CHECK]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_ASSIGN_TO_KEYCACHE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_PRELOAD_KEYS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_FLUSH]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_KILL]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_ANALYZE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_ROLLBACK]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_ROLLBACK_TO_SAVEPOINT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_COMMIT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SAVEPOINT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_RELEASE_SAVEPOINT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SLAVE_START]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SLAVE_STOP]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_BEGIN]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CHANGE_MASTER]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CHANGE_REPLICATION_FILTER]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_RENAME_TABLE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_RESET]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_PURGE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_PURGE_BEFORE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_BINLOGS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_OPEN_TABLES]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_HA_OPEN]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_HA_CLOSE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_HA_READ]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_SLAVE_HOSTS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_DELETE_MULTI]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_UPDATE_MULTI]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_BINLOG_EVENTS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_DO]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_WARNS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_EMPTY_QUERY]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_ERRORS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_STORAGE_ENGINES]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_PRIVILEGES]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_HELP]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CREATE_USER]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_DROP_USER]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_RENAME_USER]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_REVOKE_ALL]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CHECKSUM]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CREATE_PROCEDURE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CREATE_SPFUNCTION]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CALL]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_DROP_PROCEDURE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_ALTER_PROCEDURE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_ALTER_FUNCTION]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_CREATE_PROC]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_CREATE_FUNC]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_STATUS_PROC]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_STATUS_FUNC]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_PREPARE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_EXECUTE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_DEALLOCATE_PREPARE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CREATE_VIEW]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_DROP_VIEW]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CREATE_TRIGGER]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_DROP_TRIGGER]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_XA_START]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_XA_END]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_XA_PREPARE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_XA_COMMIT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_XA_ROLLBACK]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_XA_RECOVER]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_PROC_CODE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_FUNC_CODE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_ALTER_TABLESPACE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_BINLOG_BASE64_EVENT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_PLUGINS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CREATE_SERVER]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_DROP_SERVER]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_ALTER_SERVER]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_CREATE_EVENT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_ALTER_EVENT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_DROP_EVENT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_CREATE_EVENT]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_EVENTS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_CREATE_TRIGGER]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_ALTER_DB_UPGRADE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_PROFILE]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_PROFILES]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SIGNAL]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_RESIGNAL]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_RELAYLOG_EVENTS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_GET_DIAGNOSTICS]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_ALTER_USER]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_EXPLAIN_OTHER]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_SHOW_CREATE_USER]|= CF_ALLOW_PROTOCOL_PLUGIN; sql_command_flags[SQLCOM_END]|= CF_ALLOW_PROTOCOL_PLUGIN; } bool sqlcom_can_generate_row_events(enum enum_sql_command command) { return (sql_command_flags[command] & CF_CAN_GENERATE_ROW_EVENTS); } bool is_update_query(enum enum_sql_command command) { assert(command >= 0 && command <= SQLCOM_END); return (sql_command_flags[command] & CF_CHANGES_DATA) != 0; } bool is_explainable_query(enum enum_sql_command command) { assert(command >= 0 && command <= SQLCOM_END); return (sql_command_flags[command] & CF_CAN_BE_EXPLAINED) != 0; } /** Check if a sql command is allowed to write to log tables. @param command The SQL command @return true if writing is allowed */ bool is_log_table_write_query(enum enum_sql_command command) { assert(command >= 0 && command <= SQLCOM_END); return (sql_command_flags[command] & CF_WRITE_LOGS_COMMAND) != 0; } void execute_init_command(THD *thd, LEX_STRING *init_command, mysql_rwlock_t *var_lock) { Protocol_classic *protocol= thd->get_protocol_classic(); Vio* save_vio; ulong save_client_capabilities; COM_DATA com_data; mysql_rwlock_rdlock(var_lock); if (!init_command->length) { mysql_rwlock_unlock(var_lock); return; } /* copy the value under a lock, and release the lock. init_command has to be executed without a lock held, as it may try to change itself */ size_t len= init_command->length; char *buf= thd->strmake(init_command->str, len); mysql_rwlock_unlock(var_lock); #if defined(ENABLED_PROFILING) thd->profiling.start_new_query(); thd->profiling.set_query_source(buf, len); #endif THD_STAGE_INFO(thd, stage_execution_of_init_command); save_client_capabilities= protocol->get_client_capabilities(); protocol->add_client_capability(CLIENT_MULTI_QUERIES); /* We don't need return result of execution to client side. To forbid this we should set thd->net.vio to 0. */ save_vio= protocol->get_vio(); protocol->set_vio(NULL); protocol->create_command(&com_data, COM_QUERY, (uchar *) buf, len); dispatch_command(thd, &com_data, COM_QUERY); protocol->set_client_capabilities(save_client_capabilities); protocol->set_vio(save_vio); #if defined(ENABLED_PROFILING) thd->profiling.finish_current_query(); #endif } /* This works because items are allocated with sql_alloc() */ void free_items(Item *item) { Item *next; DBUG_ENTER("free_items"); for (; item ; item=next) { next=item->next; item->delete_self(); } DBUG_VOID_RETURN; } /** This works because items are allocated with sql_alloc(). @note The function also handles null pointers (empty list). */ void cleanup_items(Item *item) { DBUG_ENTER("cleanup_items"); for (; item ; item=item->next) item->cleanup(); DBUG_VOID_RETURN; } #ifndef EMBEDDED_LIBRARY /** Read one command from connection and execute it (query or simple command). This function is called in loop from thread function. For profiling to work, it must never be called recursively. @retval 0 success @retval 1 request of thread shutdown (see dispatch_command() description) */ bool do_command(THD *thd) { bool return_value; int rc; const bool classic= (thd->get_protocol()->type() == Protocol::PROTOCOL_TEXT || thd->get_protocol()->type() == Protocol::PROTOCOL_BINARY); NET *net= NULL; enum enum_server_command command; COM_DATA com_data; DBUG_ENTER("do_command"); /* indicator of uninitialized lex => normal flow of errors handling (see my_message_sql) */ thd->lex->set_current_select(0); /* XXX: this code is here only to clear possible errors of init_connect. Consider moving to prepare_new_connection_state() instead. That requires making sure the DA is cleared before non-parsing statements such as COM_QUIT. */ thd->clear_error(); // Clear error message thd->get_stmt_da()->reset_diagnostics_area(); if (classic) { /* This thread will do a blocking read from the client which will be interrupted when the next command is received from the client, the connection is closed or "net_wait_timeout" number of seconds has passed. */ net= thd->get_protocol_classic()->get_net(); my_net_set_read_timeout(net, thd->variables.net_wait_timeout); net_new_transaction(net); } /* Synchronization point for testing of KILL_CONNECTION. This sync point can wait here, to simulate slow code execution between the last test of thd->killed and blocking in read(). The goal of this test is to verify that a connection does not hang, if it is killed at this point of execution. (Bug#37780 - main.kill fails randomly) Note that the sync point wait itself will be terminated by a kill. In this case it consumes a condition broadcast, but does not change anything else. The consumed broadcast should not matter here, because the read/recv() below doesn't use it. */ DEBUG_SYNC(thd, "before_do_command_net_read"); /* Because of networking layer callbacks in place, this call will maintain the following instrumentation: - IDLE events - SOCKET events - STATEMENT events - STAGE events when reading a new network packet. In particular, a new instrumented statement is started. See init_net_server_extension() */ thd->m_server_idle= true; rc= thd->get_protocol()->get_command(&com_data, &command); thd->m_server_idle= false; if (rc) { if (classic) { DBUG_PRINT("info",("Got error %d reading command from socket %s", net->error, vio_description(net->vio))); } else { DBUG_PRINT("info",("Got error %d reading command from %s protocol", rc, (thd->get_protocol()->type() == Protocol::PROTOCOL_LOCAL) ? "local" : "plugin")); } /* Instrument this broken statement as "statement/com/error" */ thd->m_statement_psi= MYSQL_REFINE_STATEMENT(thd->m_statement_psi, com_statement_info[COM_END].m_key); /* Check if we can continue without closing the connection */ /* The error must be set. */ assert(thd->is_error()); thd->send_statement_status(); /* Mark the statement completed. */ MYSQL_END_STATEMENT(thd->m_statement_psi, thd->get_stmt_da()); thd->m_statement_psi= NULL; thd->m_digest= NULL; if (rc < 0) { return_value= TRUE; // We have to close it. goto out; } if (classic) net->error= 0; return_value= FALSE; goto out; } DBUG_PRINT("info",("Command on %s = %d (%s)", vio_description(net->vio), command, command_name[command].str)); DBUG_PRINT("info", ("packet: '%*.s'; command: %d", (int)thd->get_protocol_classic()->get_packet_length(), thd->get_protocol_classic()->get_raw_packet(), command)); if (thd->get_protocol_classic()->bad_packet) assert(0); // Should be caught earlier // Reclaim some memory thd->get_protocol_classic()->get_packet()->shrink( thd->variables.net_buffer_length); /* Restore read timeout value */ if (classic) my_net_set_read_timeout(net, thd->variables.net_read_timeout); return_value= dispatch_command(thd, &com_data, command); thd->get_protocol_classic()->get_packet()->shrink( thd->variables.net_buffer_length); out: /* The statement instrumentation must be closed in all cases. */ assert(thd->m_digest == NULL); assert(thd->m_statement_psi == NULL); DBUG_RETURN(return_value); } #endif /* EMBEDDED_LIBRARY */ /** @brief Determine if an attempt to update a non-temporary table while the read-only option was enabled has been made. This is a helper function to mysql_execute_command. @note SQLCOM_UPDATE_MULTI is an exception and delt with elsewhere. @see mysql_execute_command @returns Status code @retval TRUE The statement should be denied. @retval FALSE The statement isn't updating any relevant tables. */ static my_bool deny_updates_if_read_only_option(THD *thd, TABLE_LIST *all_tables) { DBUG_ENTER("deny_updates_if_read_only_option"); if (!check_readonly(thd, false)) DBUG_RETURN(FALSE); LEX *lex = thd->lex; if (!(sql_command_flags[lex->sql_command] & CF_CHANGES_DATA)) DBUG_RETURN(FALSE); /* Multi update is an exception and is dealt with later. */ if (lex->sql_command == SQLCOM_UPDATE_MULTI) DBUG_RETURN(FALSE); const my_bool create_temp_tables= (lex->sql_command == SQLCOM_CREATE_TABLE) && (lex->create_info.options & HA_LEX_CREATE_TMP_TABLE); const my_bool create_real_tables= (lex->sql_command == SQLCOM_CREATE_TABLE) && !(lex->create_info.options & HA_LEX_CREATE_TMP_TABLE); const my_bool drop_temp_tables= (lex->sql_command == SQLCOM_DROP_TABLE) && lex->drop_temporary; const my_bool update_real_tables= ((create_real_tables || some_non_temp_table_to_be_updated(thd, all_tables)) && !(create_temp_tables || drop_temp_tables)); const my_bool create_or_drop_databases= (lex->sql_command == SQLCOM_CREATE_DB) || (lex->sql_command == SQLCOM_DROP_DB); if (update_real_tables || create_or_drop_databases) { /* An attempt was made to modify one or more non-temporary tables. */ DBUG_RETURN(TRUE); } /* Assuming that only temporary tables are modified. */ DBUG_RETURN(FALSE); } /** Check whether max statement time is applicable to statement or not. @param thd Thread (session) context. @return true if max statement time is applicable to statement @return false otherwise. */ static inline bool is_timer_applicable_to_statement(THD *thd) { bool timer_value_is_set= (thd->lex->max_execution_time || thd->variables.max_execution_time); /** Following conditions are checked, - is SELECT statement. - timer support is implemented and it is initialized. - statement is not made by the slave threads. - timer is not set for statement - timer out value of is set - SELECT statement is not from any stored programs. */ return (thd->lex->sql_command == SQLCOM_SELECT && (have_statement_timeout == SHOW_OPTION_YES) && !thd->slave_thread && !thd->timer && timer_value_is_set && !thd->sp_runtime_ctx); } /** Get the maximum execution time for a statement. @return Length of time in milliseconds. @remark A zero timeout means that no timeout should be applied to this particular statement. */ static inline ulong get_max_execution_time(THD *thd) { return (thd->lex->max_execution_time ? thd->lex->max_execution_time : thd->variables.max_execution_time); } /** Set the time until the currently running statement is aborted. @param thd Thread (session) context. @return true if the timer was armed. */ static inline bool set_statement_timer(THD *thd) { ulong max_execution_time= get_max_execution_time(thd); /** whether timer can be set for the statement or not should be checked before calling set_statement_timer function. */ assert(is_timer_applicable_to_statement(thd) == true); assert(thd->timer == NULL); thd->timer= thd_timer_set(thd, thd->timer_cache, max_execution_time); thd->timer_cache= NULL; if (thd->timer) thd->status_var.max_execution_time_set++; else thd->status_var.max_execution_time_set_failed++; return thd->timer; } /** Deactivate the timer associated with the statement that was executed. @param thd Thread (session) context. */ void reset_statement_timer(THD *thd) { assert(thd->timer); /* Cache the timer object if it can be reused. */ thd->timer_cache= thd_timer_reset(thd->timer); thd->timer= NULL; } /** Perform one connection-level (COM_XXXX) command. @param thd connection handle @param command type of command to perform @com_data com_data union to store the generated command @todo set thd->lex->sql_command to SQLCOM_END here. @todo The following has to be changed to an 8 byte integer @retval 0 ok @retval 1 request of thread shutdown, i. e. if command is COM_QUIT/COM_SHUTDOWN */ bool dispatch_command(THD *thd, const COM_DATA *com_data, enum enum_server_command command) { bool error= 0; Global_THD_manager *thd_manager= Global_THD_manager::get_instance(); DBUG_ENTER("dispatch_command"); DBUG_PRINT("info", ("command: %d", command)); /* SHOW PROFILE instrumentation, begin */ #if defined(ENABLED_PROFILING) thd->profiling.start_new_query(); #endif /* DTRACE instrumentation, begin */ MYSQL_COMMAND_START(thd->thread_id(), command, (char *) thd->security_context()->priv_user().str, (char *) thd->security_context()->host_or_ip().str); /* Performance Schema Interface instrumentation, begin */ thd->m_statement_psi= MYSQL_REFINE_STATEMENT(thd->m_statement_psi, com_statement_info[command].m_key); thd->set_command(command); /* Commands which always take a long time are logged into the slow log only if opt_log_slow_admin_statements is set. */ thd->enable_slow_log= TRUE; thd->lex->sql_command= SQLCOM_END; /* to avoid confusing VIEW detectors */ thd->set_time(); if (thd->is_valid_time() == false) { /* If the time has gone past 2038 we need to shutdown the server. But there is possibility of getting invalid time value on some platforms. For example, gettimeofday() might return incorrect value on solaris platform. Hence validating the current time with 5 iterations before initiating the normal server shutdown process because of time getting past 2038. */ const int max_tries= 5; sql_print_warning("Current time has got past year 2038. Validating current " "time with %d iterations before initiating the normal " "server shutdown process.", max_tries); int tries= 0; while (++tries <= max_tries) { thd->set_time(); if (thd->is_valid_time() == true) { sql_print_warning("Iteration %d: Obtained valid current time from " "system", tries); break; } sql_print_warning("Iteration %d: Current time obtained from system is " "greater than 2038", tries); } if (tries > max_tries) { /* If the time has got past 2038 we need to shut this server down. We do this by making sure every command is a shutdown and we have enough privileges to shut the server down TODO: remove this when we have full 64 bit my_time_t support */ sql_print_error("This MySQL server doesn't support dates later than 2038"); ulong master_access= thd->security_context()->master_access(); thd->security_context()->set_master_access(master_access | SHUTDOWN_ACL); command= COM_SHUTDOWN; } } thd->set_query_id(next_query_id()); thd->reset_rewritten_query(); thd_manager->inc_thread_running(); if (!(server_command_flags[command] & CF_SKIP_QUESTIONS)) thd->status_var.questions++; /** Clear the set of flags that are expected to be cleared at the beginning of each command. */ thd->server_status&= ~SERVER_STATUS_CLEAR_SET; if (thd->get_protocol()->type() == Protocol::PROTOCOL_PLUGIN && !(server_command_flags[command] & CF_ALLOW_PROTOCOL_PLUGIN)) { my_error(ER_PLUGGABLE_PROTOCOL_COMMAND_NOT_SUPPORTED, MYF(0)); thd->killed= THD::KILL_CONNECTION; error= true; goto done; } /** Enforce password expiration for all RPC commands, except the following: COM_QUERY does a more fine-grained check later. COM_STMT_CLOSE and COM_STMT_SEND_LONG_DATA don't return anything. COM_PING only discloses information that the server is running, and that's available through other means. COM_QUIT should work even for expired statements. */ if (unlikely(thd->security_context()->password_expired() && command != COM_QUERY && command != COM_STMT_CLOSE && command != COM_STMT_SEND_LONG_DATA && command != COM_PING && command != COM_QUIT)) { my_error(ER_MUST_CHANGE_PASSWORD, MYF(0)); goto done; } #ifndef EMBEDDED_LIBRARY if (mysql_audit_notify(thd, AUDIT_EVENT(MYSQL_AUDIT_COMMAND_START), command, command_name[command].str)) { goto done; } #endif /* !EMBEDDED_LIBRARY */ switch (command) { case COM_INIT_DB: { LEX_STRING tmp; thd->status_var.com_stat[SQLCOM_CHANGE_DB]++; thd->convert_string(&tmp, system_charset_info, com_data->com_init_db.db_name, com_data->com_init_db.length, thd->charset()); LEX_CSTRING tmp_cstr= {tmp.str, tmp.length}; if (!mysql_change_db(thd, tmp_cstr, FALSE)) { query_logger.general_log_write(thd, command, thd->db().str, thd->db().length); my_ok(thd); } break; } #ifdef HAVE_REPLICATION case COM_REGISTER_SLAVE: { // TODO: access of protocol_classic should be removed if (!register_slave(thd, thd->get_protocol_classic()->get_raw_packet(), thd->get_protocol_classic()->get_packet_length())) my_ok(thd); break; } #endif case COM_RESET_CONNECTION: { thd->status_var.com_other++; thd->cleanup_connection(); my_ok(thd); break; } case COM_CHANGE_USER: { int auth_rc; thd->status_var.com_other++; thd->cleanup_connection(); USER_CONN *save_user_connect= const_cast<USER_CONN*>(thd->get_user_connect()); LEX_CSTRING save_db= thd->db(); Security_context save_security_ctx(*(thd->security_context())); auth_rc= acl_authenticate(thd, COM_CHANGE_USER); #ifndef EMBEDDED_LIBRARY auth_rc|= mysql_audit_notify(thd, AUDIT_EVENT(MYSQL_AUDIT_CONNECTION_CHANGE_USER)); #endif if (auth_rc) { *thd->security_context()= save_security_ctx; thd->set_user_connect(save_user_connect); thd->reset_db(save_db); my_error(ER_ACCESS_DENIED_CHANGE_USER_ERROR, MYF(0), thd->security_context()->user().str, thd->security_context()->host_or_ip().str, (thd->password ? ER(ER_YES) : ER(ER_NO))); thd->killed= THD::KILL_CONNECTION; error=true; } else { #ifndef NO_EMBEDDED_ACCESS_CHECKS /* we've authenticated new user */ if (save_user_connect) decrease_user_connections(save_user_connect); #endif /* NO_EMBEDDED_ACCESS_CHECKS */ mysql_mutex_lock(&thd->LOCK_thd_data); my_free(const_cast<char*>(save_db.str)); save_db= NULL_CSTR; mysql_mutex_unlock(&thd->LOCK_thd_data); } break; } case COM_STMT_EXECUTE: { mysqld_stmt_execute(thd, com_data->com_stmt_execute.stmt_id, com_data->com_stmt_execute.flags, com_data->com_stmt_execute.params, com_data->com_stmt_execute.params_length); break; } case COM_STMT_FETCH: { mysqld_stmt_fetch(thd, com_data->com_stmt_fetch.stmt_id, com_data->com_stmt_fetch.num_rows); break; } case COM_STMT_SEND_LONG_DATA: { mysql_stmt_get_longdata(thd, com_data->com_stmt_send_long_data.stmt_id, com_data->com_stmt_send_long_data.param_number, com_data->com_stmt_send_long_data.longdata, com_data->com_stmt_send_long_data.length); break; } case COM_STMT_PREPARE: { mysqld_stmt_prepare(thd, com_data->com_stmt_prepare.query, com_data->com_stmt_prepare.length); break; } case COM_STMT_CLOSE: { mysqld_stmt_close(thd, com_data->com_stmt_close.stmt_id); break; } case COM_STMT_RESET: { mysqld_stmt_reset(thd, com_data->com_stmt_reset.stmt_id); break; } case COM_QUERY: { assert(thd->m_digest == NULL); thd->m_digest= & thd->m_digest_state; thd->m_digest->reset(thd->m_token_array, max_digest_length); if (alloc_query(thd, com_data->com_query.query, com_data->com_query.length)) break; // fatal error is set MYSQL_QUERY_START(const_cast<char*>(thd->query().str), thd->thread_id(), (char *) (thd->db().str ? thd->db().str : ""), (char *) thd->security_context()->priv_user().str, (char *) thd->security_context()->host_or_ip().str); const char *packet_end= thd->query().str + thd->query().length; if (opt_general_log_raw) query_logger.general_log_write(thd, command, thd->query().str, thd->query().length); DBUG_PRINT("query",("%-.4096s", thd->query().str)); #if defined(ENABLED_PROFILING) thd->profiling.set_query_source(thd->query().str, thd->query().length); #endif Parser_state parser_state; if (parser_state.init(thd, thd->query().str, thd->query().length)) break; parser_state.m_input.m_has_digest = true; mysql_parse(thd, &parser_state); while (!thd->killed && (parser_state.m_lip.found_semicolon != NULL) && ! thd->is_error()) { /* Multiple queries exits, execute them individually */ const char *beginning_of_next_stmt= parser_state.m_lip.found_semicolon; /* Finalize server status flags after executing a statement. */ thd->update_server_status(); thd->send_statement_status(); query_cache.end_of_result(thd); #ifndef EMBEDDED_LIBRARY mysql_audit_notify(thd, AUDIT_EVENT(MYSQL_AUDIT_GENERAL_STATUS), thd->get_stmt_da()->is_error() ? thd->get_stmt_da()->mysql_errno() : 0, command_name[command].str, command_name[command].length); #endif size_t length= static_cast<size_t>(packet_end - beginning_of_next_stmt); log_slow_statement(thd); /* Remove garbage at start of query */ while (length > 0 && my_isspace(thd->charset(), *beginning_of_next_stmt)) { beginning_of_next_stmt++; length--; } /* PSI end */ MYSQL_END_STATEMENT(thd->m_statement_psi, thd->get_stmt_da()); thd->m_statement_psi= NULL; thd->m_digest= NULL; /* DTRACE end */ if (MYSQL_QUERY_DONE_ENABLED()) { MYSQL_QUERY_DONE(thd->is_error()); } /* SHOW PROFILE end */ #if defined(ENABLED_PROFILING) thd->profiling.finish_current_query(); #endif /* SHOW PROFILE begin */ #if defined(ENABLED_PROFILING) thd->profiling.start_new_query("continuing"); thd->profiling.set_query_source(beginning_of_next_stmt, length); #endif /* DTRACE begin */ MYSQL_QUERY_START(const_cast<char*>(beginning_of_next_stmt), thd->thread_id(), (char *) (thd->db().str ? thd->db().str : ""), (char *) thd->security_context()->priv_user().str, (char *) thd->security_context()->host_or_ip().str); /* PSI begin */ thd->m_digest= & thd->m_digest_state; thd->m_digest->reset(thd->m_token_array, max_digest_length); thd->m_statement_psi= MYSQL_START_STATEMENT(&thd->m_statement_state, com_statement_info[command].m_key, thd->db().str, thd->db().length, thd->charset(), NULL); THD_STAGE_INFO(thd, stage_starting); thd->set_query(beginning_of_next_stmt, length); thd->set_query_id(next_query_id()); /* Count each statement from the client. */ thd->status_var.questions++; thd->set_time(); /* Reset the query start time. */ parser_state.reset(beginning_of_next_stmt, length); /* TODO: set thd->lex->sql_command to SQLCOM_END here */ mysql_parse(thd, &parser_state); } /* Need to set error to true for graceful shutdown */ if((thd->lex->sql_command == SQLCOM_SHUTDOWN) && (thd->get_stmt_da()->is_ok())) error= TRUE; DBUG_PRINT("info",("query ready")); break; } case COM_FIELD_LIST: // This isn't actually needed { char *fields; /* Locked closure of all tables */ TABLE_LIST table_list; LEX_STRING table_name; LEX_STRING db; push_deprecated_warn(thd, "COM_FIELD_LIST", "SHOW COLUMNS FROM statement"); /* SHOW statements should not add the used tables to the list of tables used in a transaction. */ MDL_savepoint mdl_savepoint= thd->mdl_context.mdl_savepoint(); thd->status_var.com_stat[SQLCOM_SHOW_FIELDS]++; if (thd->copy_db_to(&db.str, &db.length)) break; thd->convert_string(&table_name, system_charset_info, (char *) com_data->com_field_list.table_name, com_data->com_field_list.table_name_length, thd->charset()); enum_ident_name_check ident_check_status= check_table_name(table_name.str, table_name.length, FALSE); if (ident_check_status == IDENT_NAME_WRONG) { /* this is OK due to convert_string() null-terminating the string */ my_error(ER_WRONG_TABLE_NAME, MYF(0), table_name.str); break; } else if (ident_check_status == IDENT_NAME_TOO_LONG) { my_error(ER_TOO_LONG_IDENT, MYF(0), table_name.str); break; } mysql_reset_thd_for_next_command(thd); lex_start(thd); /* Must be before we init the table list. */ if (lower_case_table_names) table_name.length= my_casedn_str(files_charset_info, table_name.str); table_list.init_one_table(db.str, db.length, table_name.str, table_name.length, table_name.str, TL_READ); /* Init TABLE_LIST members necessary when the undelrying table is view. */ table_list.select_lex= thd->lex->select_lex; thd->lex-> select_lex->table_list.link_in_list(&table_list, &table_list.next_local); thd->lex->add_to_query_tables(&table_list); if (is_infoschema_db(table_list.db, table_list.db_length)) { ST_SCHEMA_TABLE *schema_table= find_schema_table(thd, table_list.alias); if (schema_table) table_list.schema_table= schema_table; } if (!(fields= (char *) thd->memdup(com_data->com_field_list.query, com_data->com_field_list.query_length))) break; // Don't count end \0 thd->set_query(fields, com_data->com_field_list.query_length - 1); query_logger.general_log_print(thd, command, "%s %s", table_list.table_name, fields); if (open_temporary_tables(thd, &table_list)) break; if (check_table_access(thd, SELECT_ACL, &table_list, TRUE, UINT_MAX, FALSE)) break; /* Turn on an optimization relevant if the underlying table is a view: do not fill derived tables. */ thd->lex->sql_command= SQLCOM_SHOW_FIELDS; // See comment in opt_trace_disable_if_no_security_context_access() Opt_trace_start ots(thd, &table_list, thd->lex->sql_command, NULL, NULL, 0, NULL, NULL); mysqld_list_fields(thd,&table_list,fields); thd->lex->unit->cleanup(true); /* No need to rollback statement transaction, it's not started. */ assert(thd->get_transaction()->is_empty(Transaction_ctx::STMT)); close_thread_tables(thd); thd->mdl_context.rollback_to_savepoint(mdl_savepoint); if (thd->transaction_rollback_request) { /* Transaction rollback was requested since MDL deadlock was discovered while trying to open tables. Rollback transaction in all storage engines including binary log and release all locks. */ trans_rollback_implicit(thd); thd->mdl_context.release_transactional_locks(); } thd->cleanup_after_query(); break; } case COM_QUIT: /* We don't calculate statistics for this command */ query_logger.general_log_print(thd, command, NullS); // Don't give 'abort' message // TODO: access of protocol_classic should be removed if (thd->is_classic_protocol()) thd->get_protocol_classic()->get_net()->error= 0; thd->get_stmt_da()->disable_status(); // Don't send anything back error=TRUE; // End server break; #ifndef EMBEDDED_LIBRARY case COM_BINLOG_DUMP_GTID: // TODO: access of protocol_classic should be removed error= com_binlog_dump_gtid(thd, (char *)thd->get_protocol_classic()->get_raw_packet(), thd->get_protocol_classic()->get_packet_length()); break; case COM_BINLOG_DUMP: // TODO: access of protocol_classic should be removed error= com_binlog_dump(thd, (char*)thd->get_protocol_classic()->get_raw_packet(), thd->get_protocol_classic()->get_packet_length()); break; #endif case COM_REFRESH: { int not_used; push_deprecated_warn(thd, "COM_REFRESH", "FLUSH statement"); /* Initialize thd->lex since it's used in many base functions, such as open_tables(). Otherwise, it remains uninitialized and may cause crash during execution of COM_REFRESH. */ lex_start(thd); thd->status_var.com_stat[SQLCOM_FLUSH]++; ulong options= (ulong) com_data->com_refresh.options; if (trans_commit_implicit(thd)) break; thd->mdl_context.release_transactional_locks(); if (check_global_access(thd,RELOAD_ACL)) break; query_logger.general_log_print(thd, command, NullS); #ifndef NDEBUG bool debug_simulate= FALSE; DBUG_EXECUTE_IF("simulate_detached_thread_refresh", debug_simulate= TRUE;); if (debug_simulate) { /* Simulate a reload without a attached thread session. Provides a environment similar to that of when the server receives a SIGHUP signal and reloads caches and flushes tables. */ bool res; my_thread_set_THR_THD(NULL); res= reload_acl_and_cache(NULL, options | REFRESH_FAST, NULL, ¬_used); my_thread_set_THR_THD(thd); if (res) break; } else #endif if (reload_acl_and_cache(thd, options, (TABLE_LIST*) 0, ¬_used)) break; if (trans_commit_implicit(thd)) break; close_thread_tables(thd); thd->mdl_context.release_transactional_locks(); my_ok(thd); break; } #ifndef EMBEDDED_LIBRARY case COM_SHUTDOWN: { thd->status_var.com_other++; /* If the client is < 4.1.3, it is going to send us no argument; then packet_length is 0, packet[0] is the end 0 of the packet. Note that SHUTDOWN_DEFAULT is 0. If client is >= 4.1.3, the shutdown level is in packet[0]. */ enum mysql_enum_shutdown_level level; if (!thd->is_valid_time()) level= SHUTDOWN_DEFAULT; else level= com_data->com_shutdown.level; if(!shutdown(thd, level, command)) break; error= TRUE; break; } #endif case COM_STATISTICS: { STATUS_VAR current_global_status_var; ulong uptime; size_t length MY_ATTRIBUTE((unused)); ulonglong queries_per_second1000; char buff[250]; size_t buff_len= sizeof(buff); query_logger.general_log_print(thd, command, NullS); thd->status_var.com_stat[SQLCOM_SHOW_STATUS]++; mysql_mutex_lock(&LOCK_status); calc_sum_of_all_status(¤t_global_status_var); mysql_mutex_unlock(&LOCK_status); if (!(uptime= (ulong) (thd->start_time.tv_sec - server_start_time))) queries_per_second1000= 0; else queries_per_second1000= thd->query_id * 1000LL / uptime; length= my_snprintf(buff, buff_len - 1, "Uptime: %lu Threads: %d Questions: %lu " "Slow queries: %llu Opens: %llu Flush tables: %lu " "Open tables: %u Queries per second avg: %u.%03u", uptime, (int) thd_manager->get_thd_count(), (ulong) thd->query_id, current_global_status_var.long_query_count, current_global_status_var.opened_tables, refresh_version, table_cache_manager.cached_tables(), (uint) (queries_per_second1000 / 1000), (uint) (queries_per_second1000 % 1000)); #ifdef EMBEDDED_LIBRARY /* Store the buffer in permanent memory */ my_ok(thd, 0, 0, buff); #else // TODO: access of protocol_classic should be removed. // should be rewritten using store functions thd->get_protocol_classic()->write((uchar*) buff, length); thd->get_protocol_classic()->flush_net(); thd->get_stmt_da()->disable_status(); #endif break; } case COM_PING: thd->status_var.com_other++; my_ok(thd); // Tell client we are alive break; case COM_PROCESS_INFO: thd->status_var.com_stat[SQLCOM_SHOW_PROCESSLIST]++; push_deprecated_warn(thd, "COM_PROCESS_INFO", "SHOW PROCESSLIST statement"); if (!thd->security_context()->priv_user().str[0] && check_global_access(thd, PROCESS_ACL)) break; query_logger.general_log_print(thd, command, NullS); mysqld_list_processes( thd, thd->security_context()->check_access(PROCESS_ACL) ? NullS : thd->security_context()->priv_user().str, 0); break; case COM_PROCESS_KILL: { push_deprecated_warn(thd, "COM_PROCESS_KILL", "KILL CONNECTION/QUERY statement"); if (thd_manager->get_thread_id() & (~0xfffffffful)) my_error(ER_DATA_OUT_OF_RANGE, MYF(0), "thread_id", "mysql_kill()"); else { thd->status_var.com_stat[SQLCOM_KILL]++; sql_kill(thd, com_data->com_kill.id, false); } break; } case COM_SET_OPTION: { thd->status_var.com_stat[SQLCOM_SET_OPTION]++; switch (com_data->com_set_option.opt_command) { case (int) MYSQL_OPTION_MULTI_STATEMENTS_ON: //TODO: access of protocol_classic should be removed thd->get_protocol_classic()->add_client_capability( CLIENT_MULTI_STATEMENTS); my_eof(thd); break; case (int) MYSQL_OPTION_MULTI_STATEMENTS_OFF: thd->get_protocol_classic()->remove_client_capability( CLIENT_MULTI_STATEMENTS); my_eof(thd); break; default: my_message(ER_UNKNOWN_COM_ERROR, ER(ER_UNKNOWN_COM_ERROR), MYF(0)); break; } break; } case COM_DEBUG: thd->status_var.com_other++; if (check_global_access(thd, SUPER_ACL)) break; /* purecov: inspected */ mysql_print_status(); query_logger.general_log_print(thd, command, NullS); my_eof(thd); break; case COM_SLEEP: case COM_CONNECT: // Impossible here case COM_TIME: // Impossible from client case COM_DELAYED_INSERT: // INSERT DELAYED has been removed. case COM_END: default: my_message(ER_UNKNOWN_COM_ERROR, ER(ER_UNKNOWN_COM_ERROR), MYF(0)); break; } done: assert(thd->derived_tables == NULL && (thd->open_tables == NULL || (thd->locked_tables_mode == LTM_LOCK_TABLES))); /* Finalize server status flags after executing a command. */ thd->update_server_status(); if (thd->killed) thd->send_kill_message(); thd->send_statement_status(); thd->rpl_thd_ctx.session_gtids_ctx().notify_after_response_packet(thd); query_cache.end_of_result(thd); #ifndef EMBEDDED_LIBRARY if (!thd->is_error() && !thd->killed_errno()) mysql_audit_notify(thd, AUDIT_EVENT(MYSQL_AUDIT_GENERAL_RESULT), 0, NULL, 0); mysql_audit_notify(thd, AUDIT_EVENT(MYSQL_AUDIT_GENERAL_STATUS), thd->get_stmt_da()->is_error() ? thd->get_stmt_da()->mysql_errno() : 0, command_name[command].str, command_name[command].length); /* command_end is informational only. The plugin cannot abort execution of the command at thie point. */ mysql_audit_notify(thd, AUDIT_EVENT(MYSQL_AUDIT_COMMAND_END), command, command_name[command].str); #endif log_slow_statement(thd); THD_STAGE_INFO(thd, stage_cleaning_up); thd->reset_query(); thd->set_command(COM_SLEEP); thd->proc_info= 0; thd->lex->sql_command= SQLCOM_END; /* Performance Schema Interface instrumentation, end */ MYSQL_END_STATEMENT(thd->m_statement_psi, thd->get_stmt_da()); thd->m_statement_psi= NULL; thd->m_digest= NULL; thd->reset_query_for_display(); /* Prevent rewritten query from getting "stuck" in SHOW PROCESSLIST. */ thd->reset_rewritten_query(); thd_manager->dec_thread_running(); free_root(thd->mem_root,MYF(MY_KEEP_PREALLOC)); /* DTRACE instrumentation, end */ if (MYSQL_QUERY_DONE_ENABLED() && command == COM_QUERY) { MYSQL_QUERY_DONE(thd->is_error()); } if (MYSQL_COMMAND_DONE_ENABLED()) { MYSQL_COMMAND_DONE(thd->is_error()); } /* SHOW PROFILE instrumentation, end */ #if defined(ENABLED_PROFILING) thd->profiling.finish_current_query(); #endif DBUG_RETURN(error); } /** Shutdown the mysqld server. @param thd Thread (session) context. @param level Shutdown level. @param command type of command to perform @retval true success @retval false When user has insufficient privilege or unsupported shutdown level */ #ifndef EMBEDDED_LIBRARY bool shutdown(THD *thd, enum mysql_enum_shutdown_level level, enum enum_server_command command) { DBUG_ENTER("shutdown"); bool res= FALSE; thd->lex->no_write_to_binlog= 1; if (check_global_access(thd,SHUTDOWN_ACL)) goto error; /* purecov: inspected */ if (level == SHUTDOWN_DEFAULT) level= SHUTDOWN_WAIT_ALL_BUFFERS; // soon default will be configurable else if (level != SHUTDOWN_WAIT_ALL_BUFFERS) { my_error(ER_NOT_SUPPORTED_YET, MYF(0), "this shutdown level"); goto error;; } if(command == COM_SHUTDOWN) my_eof(thd); else if(command == COM_QUERY) my_ok(thd); else { my_error(ER_NOT_SUPPORTED_YET, MYF(0), "shutdown from this server command"); goto error; } DBUG_PRINT("quit",("Got shutdown command for level %u", level)); query_logger.general_log_print(thd, command, NullS); kill_mysql(); res= TRUE; error: DBUG_RETURN(res); } #endif /** Create a TABLE_LIST object for an INFORMATION_SCHEMA table. This function is used in the parser to convert a SHOW or DESCRIBE table_name command to a SELECT from INFORMATION_SCHEMA. It prepares a SELECT_LEX and a TABLE_LIST object to represent the given command as a SELECT parse tree. @param thd thread handle @param lex current lex @param table_ident table alias if it's used @param schema_table_idx the type of the INFORMATION_SCHEMA table to be created @note Due to the way this function works with memory and LEX it cannot be used outside the parser (parse tree transformations outside the parser break PS and SP). @retval 0 success @retval 1 out of memory or SHOW commands are not allowed in this version of the server. */ int prepare_schema_table(THD *thd, LEX *lex, Table_ident *table_ident, enum enum_schema_tables schema_table_idx) { SELECT_LEX *schema_select_lex= NULL; DBUG_ENTER("prepare_schema_table"); switch (schema_table_idx) { case SCH_SCHEMATA: break; case SCH_TABLE_NAMES: case SCH_TABLES: case SCH_VIEWS: case SCH_TRIGGERS: case SCH_EVENTS: { LEX_STRING db; size_t dummy; if (lex->select_lex->db == NULL && lex->copy_db_to(&lex->select_lex->db, &dummy)) DBUG_RETURN(1); if ((schema_select_lex= lex->new_empty_query_block()) == NULL) DBUG_RETURN(1); /* purecov: inspected */ db.str= schema_select_lex->db= lex->select_lex->db; schema_select_lex->table_list.first= NULL; db.length= strlen(db.str); if (check_and_convert_db_name(&db, FALSE) != IDENT_NAME_OK) DBUG_RETURN(1); break; } case SCH_COLUMNS: case SCH_STATISTICS: { assert(table_ident); TABLE_LIST **query_tables_last= lex->query_tables_last; if ((schema_select_lex= lex->new_empty_query_block()) == NULL) DBUG_RETURN(1); /* purecov: inspected */ if (!schema_select_lex->add_table_to_list(thd, table_ident, 0, 0, TL_READ, MDL_SHARED_READ)) DBUG_RETURN(1); lex->query_tables_last= query_tables_last; break; } case SCH_PROFILES: /* Mark this current profiling record to be discarded. We don't wish to have SHOW commands show up in profiling. */ #if defined(ENABLED_PROFILING) thd->profiling.discard_current_query(); #endif break; case SCH_OPTIMIZER_TRACE: case SCH_OPEN_TABLES: case SCH_VARIABLES: case SCH_STATUS: case SCH_PROCEDURES: case SCH_CHARSETS: case SCH_ENGINES: case SCH_COLLATIONS: case SCH_COLLATION_CHARACTER_SET_APPLICABILITY: case SCH_USER_PRIVILEGES: case SCH_SCHEMA_PRIVILEGES: case SCH_TABLE_PRIVILEGES: case SCH_COLUMN_PRIVILEGES: case SCH_TABLE_CONSTRAINTS: case SCH_KEY_COLUMN_USAGE: default: break; } SELECT_LEX *select_lex= lex->current_select(); if (make_schema_select(thd, select_lex, schema_table_idx)) { DBUG_RETURN(1); } TABLE_LIST *table_list= select_lex->table_list.first; table_list->schema_select_lex= schema_select_lex; table_list->schema_table_reformed= 1; DBUG_RETURN(0); } /** Read query from packet and store in thd->query. Used in COM_QUERY and COM_STMT_PREPARE. Sets the following THD variables: - query - query_length @retval FALSE ok @retval TRUE error; In this case thd->fatal_error is set */ bool alloc_query(THD *thd, const char *packet, size_t packet_length) { /* Remove garbage at start and end of query */ while (packet_length > 0 && my_isspace(thd->charset(), packet[0])) { packet++; packet_length--; } const char *pos= packet + packet_length; // Point at end null while (packet_length > 0 && (pos[-1] == ';' || my_isspace(thd->charset() ,pos[-1]))) { pos--; packet_length--; } char *query= static_cast<char*>(thd->alloc(packet_length + 1)); if (!query) return TRUE; memcpy(query, packet, packet_length); query[packet_length]= '\0'; thd->set_query(query, packet_length); /* Reclaim some memory */ if(thd->get_protocol()->type() == Protocol::PROTOCOL_TEXT || thd->get_protocol()->type() == Protocol::PROTOCOL_BINARY) thd->convert_buffer.shrink(thd->variables.net_buffer_length); return FALSE; } static bool sp_process_definer(THD *thd) { DBUG_ENTER("sp_process_definer"); LEX *lex= thd->lex; /* If the definer is not specified, this means that CREATE-statement missed DEFINER-clause. DEFINER-clause can be missed in two cases: - The user submitted a statement w/o the clause. This is a normal case, we should assign CURRENT_USER as definer. - Our slave received an updated from the master, that does not replicate definer for stored rountines. We should also assign CURRENT_USER as definer here, but also we should mark this routine as NON-SUID. This is essential for the sake of backward compatibility. The problem is the slave thread is running under "special" user (@), that actually does not exist. In the older versions we do not fail execution of a stored routine if its definer does not exist and continue the execution under the authorization of the invoker (BUG#13198). And now if we try to switch to slave-current-user (@), we will fail. Actually, this leads to the inconsistent state of master and slave (different definers, different SUID behaviour), but it seems, this is the best we can do. */ if (!lex->definer) { Prepared_stmt_arena_holder ps_arena_holder(thd); lex->definer= create_default_definer(thd); /* Error has been already reported. */ if (lex->definer == NULL) DBUG_RETURN(TRUE); if (thd->slave_thread && lex->sphead) lex->sphead->m_chistics->suid= SP_IS_NOT_SUID; } else { /* If the specified definer differs from the current user, we should check that the current user has SUPER privilege (in order to create a stored routine under another user one must have SUPER privilege). */ if ((strcmp(lex->definer->user.str, thd->security_context()->priv_user().str) || my_strcasecmp(system_charset_info, lex->definer->host.str, thd->security_context()->priv_host().str)) && check_global_access(thd, SUPER_ACL)) { my_error(ER_SPECIFIC_ACCESS_DENIED_ERROR, MYF(0), "SUPER"); DBUG_RETURN(TRUE); } } /* Check that the specified definer exists. Emit a warning if not. */ #ifndef NO_EMBEDDED_ACCESS_CHECKS if (!is_acl_user(lex->definer->host.str, lex->definer->user.str)) { push_warning_printf(thd, Sql_condition::SL_NOTE, ER_NO_SUCH_USER, ER(ER_NO_SUCH_USER), lex->definer->user.str, lex->definer->host.str); } #endif /* NO_EMBEDDED_ACCESS_CHECKS */ DBUG_RETURN(FALSE); } /** Auxiliary call that opens and locks tables for LOCK TABLES statement and initializes the list of locked tables. @param thd Thread context. @param tables List of tables to be locked. @return FALSE in case of success, TRUE in case of error. */ static bool lock_tables_open_and_lock_tables(THD *thd, TABLE_LIST *tables) { Lock_tables_prelocking_strategy lock_tables_prelocking_strategy; MDL_deadlock_and_lock_abort_error_handler deadlock_handler; MDL_savepoint mdl_savepoint= thd->mdl_context.mdl_savepoint(); uint counter; TABLE_LIST *table; thd->in_lock_tables= 1; retry: if (open_tables(thd, &tables, &counter, 0, &lock_tables_prelocking_strategy)) goto err; deadlock_handler.init(); thd->push_internal_handler(&deadlock_handler); for (table= tables; table; table= table->next_global) { if (!table->is_placeholder()) { if (table->table->s->tmp_table) { /* We allow to change temporary tables even if they were locked for read by LOCK TABLES. To avoid a discrepancy between lock acquired at LOCK TABLES time and by the statement which is later executed under LOCK TABLES we ensure that for temporary tables we always request a write lock (such discrepancy can cause problems for the storage engine). We don't set TABLE_LIST::lock_type in this case as this might result in extra warnings from THD::decide_logging_format() even though binary logging is totally irrelevant for LOCK TABLES. */ table->table->reginfo.lock_type= TL_WRITE; } else if (table->lock_type == TL_READ && ! table->prelocking_placeholder && table->table->file->ha_table_flags() & HA_NO_READ_LOCAL_LOCK) { /* In case when LOCK TABLE ... READ LOCAL was issued for table with storage engine which doesn't support READ LOCAL option and doesn't use THR_LOCK locks we need to upgrade weak SR metadata lock acquired in open_tables() to stronger SRO metadata lock. This is not needed for tables used through stored routines or triggers as we always acquire SRO (or even stronger SNRW) metadata lock for them. */ bool result= thd->mdl_context.upgrade_shared_lock( table->table->mdl_ticket, MDL_SHARED_READ_ONLY, thd->variables.lock_wait_timeout); if (deadlock_handler.need_reopen()) { /* Deadlock occurred during upgrade of metadata lock. Let us restart acquring and opening tables for LOCK TABLES. */ thd->pop_internal_handler(); close_tables_for_reopen(thd, &tables, mdl_savepoint); if (open_temporary_tables(thd, tables)) goto err; goto retry; } if (result) { thd->pop_internal_handler(); goto err; } } } } thd->pop_internal_handler(); if (lock_tables(thd, tables, counter, 0) || thd->locked_tables_list.init_locked_tables(thd)) goto err; thd->in_lock_tables= 0; return FALSE; err: thd->in_lock_tables= 0; trans_rollback_stmt(thd); /* Need to end the current transaction, so the storage engine (InnoDB) can free its locks if LOCK TABLES locked some tables before finding that it can't lock a table in its list */ trans_rollback(thd); /* Close tables and release metadata locks. */ close_thread_tables(thd); assert(!thd->locked_tables_mode); thd->mdl_context.release_transactional_locks(); return TRUE; } /** This is a wrapper for MYSQL_BIN_LOG::gtid_end_transaction. For normal statements, the function gtid_end_transaction is called in the commit handler. However, if the statement is filtered out or not written to the binary log, the commit handler is not invoked. Therefore, this wrapper calls gtid_end_transaction in case the current statement is committing but was not written to the binary log. (The function gtid_end_transaction ensures that gtid-related end-of-transaction operations are performed; this includes generating an empty transaction and calling Gtid_state::update_gtids_impl.) @param thd Thread (session) context. */ static inline void binlog_gtid_end_transaction(THD *thd) { DBUG_ENTER("binlog_gtid_end_transaction"); /* This performs end-of-transaction actions needed by GTIDs: in particular, it generates an empty transaction if needed (e.g., if the statement was filtered out). It is executed at the end of an implicitly or explicitly committing statement. In addition, it is executed after CREATE TEMPORARY TABLE or DROP TEMPORARY TABLE when they occur outside transactional context. When enforce_gtid_consistency is enabled, these statements cannot occur in transactional context, and then they behave exactly as implicitly committing: they are written to the binary log immediately, not wrapped in BEGIN/COMMIT, and cannot be rolled back. However, they do not count as implicitly committing according to stmt_causes_implicit_commit(), so we need to add special cases in the condition below. Hence the clauses for SQLCOM_CREATE_TABLE and SQLCOM_DROP_TABLE. If enforce_gtid_consistency=off, CREATE TEMPORARY TABLE and DROP TEMPORARY TABLE can occur in the middle of a transaction. Then they do not behave as DDL; they are written to the binary log inside BEGIN/COMMIT. (For base tables, SQLCOM_[CREATE|DROP]_TABLE match both the stmt_causes_implicit_commit(...) clause and the thd->lex->sql_command == SQLCOM_* clause; for temporary tables they match only thd->lex->sql_command == SQLCOM_*.) */ if ((thd->lex->sql_command == SQLCOM_COMMIT || (thd->slave_thread && (thd->lex->sql_command == SQLCOM_XA_COMMIT || thd->lex->sql_command == SQLCOM_XA_ROLLBACK)) || stmt_causes_implicit_commit(thd, CF_IMPLICIT_COMMIT_END) || ((thd->lex->sql_command == SQLCOM_CREATE_TABLE || thd->lex->sql_command == SQLCOM_DROP_TABLE) && !thd->in_multi_stmt_transaction_mode()))) (void) mysql_bin_log.gtid_end_transaction(thd); DBUG_VOID_RETURN; } /** Execute command saved in thd and lex->sql_command. @param thd Thread handle @todo - Invalidate the table in the query cache if something changed after unlocking when changes become visible. @todo: this is workaround. right way will be move invalidating in the unlock procedure. - TODO: use check_change_password() @retval FALSE OK @retval TRUE Error */ int mysql_execute_command(THD *thd, bool first_level) { int res= FALSE; LEX *const lex= thd->lex; /* first SELECT_LEX (have special meaning for many of non-SELECTcommands) */ SELECT_LEX *const select_lex= lex->select_lex; /* first table of first SELECT_LEX */ TABLE_LIST *const first_table= select_lex->get_table_list(); /* list of all tables in query */ TABLE_LIST *all_tables; /* most outer SELECT_LEX_UNIT of query */ SELECT_LEX_UNIT *const unit= lex->unit; // keep GTID violation state in order to roll it back on statement failure bool gtid_consistency_violation_state = thd->has_gtid_consistency_violation; assert(select_lex->master_unit() == unit); DBUG_ENTER("mysql_execute_command"); /* EXPLAIN OTHER isn't explainable command, but can have describe flag. */ assert(!lex->describe || is_explainable_query(lex->sql_command) || lex->sql_command == SQLCOM_EXPLAIN_OTHER); thd->work_part_info= 0; assert(thd->get_transaction()->is_empty(Transaction_ctx::STMT) || thd->in_sub_stmt); /* Each statement or replication event which might produce deadlock should handle transaction rollback on its own. So by the start of the next statement transaction rollback request should be fulfilled already. */ assert(! thd->transaction_rollback_request || thd->in_sub_stmt); /* In many cases first table of main SELECT_LEX have special meaning => check that it is first table in global list and relink it first in queries_tables list if it is necessary (we need such relinking only for queries with subqueries in select list, in this case tables of subqueries will go to global list first) all_tables will differ from first_table only if most upper SELECT_LEX do not contain tables. Because of above in place where should be at least one table in most outer SELECT_LEX we have following check: assert(first_table == all_tables); assert(first_table == all_tables && first_table != 0); */ lex->first_lists_tables_same(); /* should be assigned after making first tables same */ all_tables= lex->query_tables; /* set context for commands which do not use setup_tables */ select_lex->context.resolve_in_table_list_only(select_lex->get_table_list()); thd->get_stmt_da()->reset_diagnostics_area(); if ((thd->lex->keep_diagnostics != DA_KEEP_PARSE_ERROR) && (thd->lex->keep_diagnostics != DA_KEEP_DIAGNOSTICS)) { /* No parse errors, and it's not a diagnostic statement: remove the sql conditions from the DA! For diagnostic statements we need to keep the conditions around so we can inspec them. */ thd->get_stmt_da()->reset_condition_info(thd); } #ifdef HAVE_REPLICATION if (unlikely(thd->slave_thread)) { if (!check_database_filters(thd, thd->db().str, lex->sql_command)) { binlog_gtid_end_transaction(thd); DBUG_RETURN(0); } if (lex->sql_command == SQLCOM_DROP_TRIGGER) { /* When dropping a trigger, we need to load its table name before checking slave filter rules. */ add_table_for_trigger(thd, lex->spname->m_db, lex->spname->m_name, true, &all_tables); if (!all_tables) { /* If table name cannot be loaded, it means the trigger does not exists possibly because CREATE TRIGGER was previously skipped for this trigger according to slave filtering rules. Returning success without producing any errors in this case. */ binlog_gtid_end_transaction(thd); DBUG_RETURN(0); } // force searching in slave.cc:tables_ok() all_tables->updating= 1; } /* For fix of BUG#37051, the master stores the table map for update in the Query_log_event, and the value is assigned to thd->variables.table_map_for_update before executing the update query. If thd->variables.table_map_for_update is set, then we are replicating from a new master, we can use this value to apply filter rules without opening all the tables. However If thd->variables.table_map_for_update is not set, then we are replicating from an old master, so we just skip this and continue with the old method. And of course, the bug would still exist for old masters. */ if (lex->sql_command == SQLCOM_UPDATE_MULTI && thd->table_map_for_update) { table_map table_map_for_update= thd->table_map_for_update; uint nr= 0; TABLE_LIST *table; for (table=all_tables; table; table=table->next_global, nr++) { if (table_map_for_update & ((table_map)1 << nr)) table->updating= TRUE; else table->updating= FALSE; } if (all_tables_not_ok(thd, all_tables)) { /* we warn the slave SQL thread */ my_message(ER_SLAVE_IGNORED_TABLE, ER(ER_SLAVE_IGNORED_TABLE), MYF(0)); binlog_gtid_end_transaction(thd); DBUG_RETURN(0); } for (table=all_tables; table; table=table->next_global) table->updating= TRUE; } /* Check if statment should be skipped because of slave filtering rules Exceptions are: - UPDATE MULTI: For this statement, we want to check the filtering rules later in the code - SET: we always execute it (Not that many SET commands exists in the binary log anyway -- only 4.1 masters write SET statements, in 5.0 there are no SET statements in the binary log) - DROP TEMPORARY TABLE IF EXISTS: we always execute it (otherwise we have stale files on slave caused by exclusion of one tmp table). */ if (!(lex->sql_command == SQLCOM_UPDATE_MULTI) && !(lex->sql_command == SQLCOM_SET_OPTION) && !(lex->sql_command == SQLCOM_DROP_TABLE && lex->drop_temporary && lex->drop_if_exists) && all_tables_not_ok(thd, all_tables)) { /* we warn the slave SQL thread */ my_message(ER_SLAVE_IGNORED_TABLE, ER(ER_SLAVE_IGNORED_TABLE), MYF(0)); binlog_gtid_end_transaction(thd); DBUG_RETURN(0); } /* Execute deferred events first */ if (slave_execute_deferred_events(thd)) DBUG_RETURN(-1); } else { #endif /* HAVE_REPLICATION */ /* When option readonly is set deny operations which change non-temporary tables. Except for the replication thread and the 'super' users. */ if (deny_updates_if_read_only_option(thd, all_tables)) { err_readonly(thd); DBUG_RETURN(-1); } #ifdef HAVE_REPLICATION } /* endif unlikely slave */ #endif thd->status_var.com_stat[lex->sql_command]++; Opt_trace_start ots(thd, all_tables, lex->sql_command, &lex->var_list, thd->query().str, thd->query().length, NULL, thd->variables.character_set_client); Opt_trace_object trace_command(&thd->opt_trace); Opt_trace_array trace_command_steps(&thd->opt_trace, "steps"); assert(thd->get_transaction()->cannot_safely_rollback( Transaction_ctx::STMT) == false); switch (gtid_pre_statement_checks(thd)) { case GTID_STATEMENT_EXECUTE: break; case GTID_STATEMENT_CANCEL: DBUG_RETURN(-1); case GTID_STATEMENT_SKIP: my_ok(thd); binlog_gtid_end_transaction(thd); DBUG_RETURN(0); } /* End a active transaction so that this command will have it's own transaction and will also sync the binary log. If a DDL is not run in it's own transaction it may simply never appear on the slave in case the outside transaction rolls back. */ if (stmt_causes_implicit_commit(thd, CF_IMPLICIT_COMMIT_BEGIN)) { /* Note that this should never happen inside of stored functions or triggers as all such statements prohibited there. */ assert(! thd->in_sub_stmt); /* Statement transaction still should not be started. */ assert(thd->get_transaction()->is_empty(Transaction_ctx::STMT)); /* Implicit commit is not allowed with an active XA transaction. In this case we should not release metadata locks as the XA transaction will not be rolled back. Therefore we simply return here. */ if (trans_check_state(thd)) DBUG_RETURN(-1); /* Commit the normal transaction if one is active. */ if (trans_commit_implicit(thd)) goto error; /* Release metadata locks acquired in this transaction. */ thd->mdl_context.release_transactional_locks(); } DEBUG_SYNC(thd, "after_implicit_pre_commit"); if (gtid_pre_statement_post_implicit_commit_checks(thd)) DBUG_RETURN(-1); #ifndef EMBEDDED_LIBRARY if (mysql_audit_notify(thd, first_level ? MYSQL_AUDIT_QUERY_START : MYSQL_AUDIT_QUERY_NESTED_START, first_level ? "MYSQL_AUDIT_QUERY_START" : "MYSQL_AUDIT_QUERY_NESTED_START")) { DBUG_RETURN(1); } #endif /* !EMBEDDED_LIBRARY */ #ifndef NDEBUG if (lex->sql_command != SQLCOM_SET_OPTION) DEBUG_SYNC(thd,"before_execute_sql_command"); #endif /* Check if we are in a read-only transaction and we're trying to execute a statement which should always be disallowed in such cases. Note that this check is done after any implicit commits. */ if (thd->tx_read_only && (sql_command_flags[lex->sql_command] & CF_DISALLOW_IN_RO_TRANS)) { my_error(ER_CANT_EXECUTE_IN_READ_ONLY_TRANSACTION, MYF(0)); goto error; } /* Close tables open by HANDLERs before executing DDL statement which is going to affect those tables. This should happen before temporary tables are pre-opened as otherwise we will get errors about attempt to re-open tables if table to be changed is open through HANDLER. Note that even although this is done before any privilege checks there is no security problem here as closing open HANDLER doesn't require any privileges anyway. */ if (sql_command_flags[lex->sql_command] & CF_HA_CLOSE) mysql_ha_rm_tables(thd, all_tables); /* Check that the command is allowed on the PROTOCOL_PLUGIN */ if (thd->get_protocol()->type() == Protocol::PROTOCOL_PLUGIN && !(sql_command_flags[lex->sql_command] & CF_ALLOW_PROTOCOL_PLUGIN)) { my_error(ER_PLUGGABLE_PROTOCOL_COMMAND_NOT_SUPPORTED, MYF(0)); goto error; } /* Pre-open temporary tables to simplify privilege checking for statements which need this. */ if (sql_command_flags[lex->sql_command] & CF_PREOPEN_TMP_TABLES) { if (open_temporary_tables(thd, all_tables)) goto error; } // Save original info for EXPLAIN FOR CONNECTION if (!thd->in_sub_stmt) thd->query_plan.set_query_plan(lex->sql_command, lex, !thd->stmt_arena->is_conventional()); switch (lex->sql_command) { case SQLCOM_SHOW_STATUS: { system_status_var old_status_var= thd->status_var; thd->initial_status_var= &old_status_var; if (!(res= select_precheck(thd, lex, all_tables, first_table))) res= execute_sqlcom_select(thd, all_tables); /* Don't log SHOW STATUS commands to slow query log */ thd->server_status&= ~(SERVER_QUERY_NO_INDEX_USED | SERVER_QUERY_NO_GOOD_INDEX_USED); /* restore status variables, as we don't want 'show status' to cause changes */ mysql_mutex_lock(&LOCK_status); add_diff_to_status(&global_status_var, &thd->status_var, &old_status_var); thd->status_var= old_status_var; thd->initial_status_var= NULL; mysql_mutex_unlock(&LOCK_status); break; } case SQLCOM_SHOW_EVENTS: #ifdef EMBEDDED_LIBRARY my_error(ER_NOT_SUPPORTED_YET, MYF(0), "embedded server"); break; #endif case SQLCOM_SHOW_STATUS_PROC: case SQLCOM_SHOW_STATUS_FUNC: case SQLCOM_SHOW_DATABASES: case SQLCOM_SHOW_TABLES: case SQLCOM_SHOW_TRIGGERS: case SQLCOM_SHOW_TABLE_STATUS: case SQLCOM_SHOW_OPEN_TABLES: case SQLCOM_SHOW_PLUGINS: case SQLCOM_SHOW_FIELDS: case SQLCOM_SHOW_KEYS: case SQLCOM_SHOW_VARIABLES: case SQLCOM_SHOW_CHARSETS: case SQLCOM_SHOW_COLLATIONS: case SQLCOM_SHOW_STORAGE_ENGINES: case SQLCOM_SHOW_PROFILE: case SQLCOM_SELECT: { DBUG_EXECUTE_IF("use_attachable_trx", thd->begin_attachable_ro_transaction();); thd->clear_current_query_costs(); res= select_precheck(thd, lex, all_tables, first_table); if (!res) res= execute_sqlcom_select(thd, all_tables); thd->save_current_query_costs(); DBUG_EXECUTE_IF("use_attachable_trx", thd->end_attachable_transaction();); break; } case SQLCOM_PREPARE: { mysql_sql_stmt_prepare(thd); break; } case SQLCOM_EXECUTE: { mysql_sql_stmt_execute(thd); break; } case SQLCOM_DEALLOCATE_PREPARE: { mysql_sql_stmt_close(thd); break; } case SQLCOM_DO: res= mysql_do(thd, lex); break; case SQLCOM_EMPTY_QUERY: my_ok(thd); break; case SQLCOM_HELP: res= mysqld_help(thd,lex->help_arg); break; #ifndef EMBEDDED_LIBRARY case SQLCOM_PURGE: { if (check_global_access(thd, SUPER_ACL)) goto error; /* PURGE MASTER LOGS TO 'file' */ res = purge_master_logs(thd, lex->to_log); break; } case SQLCOM_PURGE_BEFORE: { Item *it; if (check_global_access(thd, SUPER_ACL)) goto error; /* PURGE MASTER LOGS BEFORE 'data' */ it= lex->purge_value_list.head(); if ((!it->fixed && it->fix_fields(lex->thd, &it)) || it->check_cols(1)) { my_error(ER_WRONG_ARGUMENTS, MYF(0), "PURGE LOGS BEFORE"); goto error; } it= new Item_func_unix_timestamp(it); /* it is OK only emulate fix_fieds, because we need only value of constant */ it->quick_fix_field(); time_t purge_time= static_cast<time_t>(it->val_int()); if (thd->is_error()) goto error; res = purge_master_logs_before_date(thd, purge_time); break; } #endif case SQLCOM_SHOW_WARNS: { res= mysqld_show_warnings(thd, (ulong) ((1L << (uint) Sql_condition::SL_NOTE) | (1L << (uint) Sql_condition::SL_WARNING) | (1L << (uint) Sql_condition::SL_ERROR) )); break; } case SQLCOM_SHOW_ERRORS: { res= mysqld_show_warnings(thd, (ulong) (1L << (uint) Sql_condition::SL_ERROR)); break; } case SQLCOM_SHOW_PROFILES: { #if defined(ENABLED_PROFILING) thd->profiling.discard_current_query(); res= thd->profiling.show_profiles(); if (res) goto error; #else my_error(ER_FEATURE_DISABLED, MYF(0), "SHOW PROFILES", "enable-profiling"); goto error; #endif break; } #ifdef HAVE_REPLICATION case SQLCOM_SHOW_SLAVE_HOSTS: { if (check_global_access(thd, REPL_SLAVE_ACL)) goto error; res= show_slave_hosts(thd); break; } case SQLCOM_SHOW_RELAYLOG_EVENTS: { if (check_global_access(thd, REPL_SLAVE_ACL)) goto error; res = mysql_show_relaylog_events(thd); break; } case SQLCOM_SHOW_BINLOG_EVENTS: { if (check_global_access(thd, REPL_SLAVE_ACL)) goto error; res = mysql_show_binlog_events(thd); break; } #endif case SQLCOM_ASSIGN_TO_KEYCACHE: { assert(first_table == all_tables && first_table != 0); if (check_access(thd, INDEX_ACL, first_table->db, &first_table->grant.privilege, &first_table->grant.m_internal, 0, 0)) goto error; res= mysql_assign_to_keycache(thd, first_table, &lex->ident); break; } case SQLCOM_PRELOAD_KEYS: { assert(first_table == all_tables && first_table != 0); if (check_access(thd, INDEX_ACL, first_table->db, &first_table->grant.privilege, &first_table->grant.m_internal, 0, 0)) goto error; res = mysql_preload_keys(thd, first_table); break; } #ifdef HAVE_REPLICATION case SQLCOM_CHANGE_MASTER: { if (check_global_access(thd, SUPER_ACL)) goto error; res= change_master_cmd(thd); break; } case SQLCOM_SHOW_SLAVE_STAT: { /* Accept one of two privileges */ if (check_global_access(thd, SUPER_ACL | REPL_CLIENT_ACL)) goto error; res= show_slave_status_cmd(thd); break; } case SQLCOM_SHOW_MASTER_STAT: { /* Accept one of two privileges */ if (check_global_access(thd, SUPER_ACL | REPL_CLIENT_ACL)) goto error; res = show_master_status(thd); break; } #endif /* HAVE_REPLICATION */ case SQLCOM_SHOW_ENGINE_STATUS: { if (check_global_access(thd, PROCESS_ACL)) goto error; res = ha_show_status(thd, lex->create_info.db_type, HA_ENGINE_STATUS); break; } case SQLCOM_SHOW_ENGINE_MUTEX: { if (check_global_access(thd, PROCESS_ACL)) goto error; res = ha_show_status(thd, lex->create_info.db_type, HA_ENGINE_MUTEX); break; } case SQLCOM_CREATE_TABLE: { assert(first_table == all_tables && first_table != 0); bool link_to_local; TABLE_LIST *create_table= first_table; TABLE_LIST *select_tables= lex->create_last_non_select_table->next_global; /* Code below (especially in mysql_create_table() and Query_result_create methods) may modify HA_CREATE_INFO structure in LEX, so we have to use a copy of this structure to make execution prepared statement- safe. A shallow copy is enough as this code won't modify any memory referenced from this structure. */ HA_CREATE_INFO create_info(lex->create_info); /* We need to copy alter_info for the same reasons of re-execution safety, only in case of Alter_info we have to do (almost) a deep copy. */ Alter_info alter_info(lex->alter_info, thd->mem_root); if (thd->is_fatal_error) { /* If out of memory when creating a copy of alter_info. */ res= 1; goto end_with_restore_list; } if (((lex->create_info.used_fields & HA_CREATE_USED_DATADIR) != 0 || (lex->create_info.used_fields & HA_CREATE_USED_INDEXDIR) != 0) && check_access(thd, FILE_ACL, any_db, NULL, NULL, FALSE, FALSE)) { res= 1; my_error(ER_SPECIFIC_ACCESS_DENIED_ERROR, MYF(0), "FILE"); goto end_with_restore_list; } if ((res= create_table_precheck(thd, select_tables, create_table))) goto end_with_restore_list; /* Might have been updated in create_table_precheck */ create_info.alias= create_table->alias; /* Assign target tablespace name to enable locking in lock_table_names(). Reject invalid names. */ if (create_info.tablespace) { if (check_tablespace_name(create_info.tablespace) != IDENT_NAME_OK) goto end_with_restore_list; if (!thd->make_lex_string(&create_table->target_tablespace_name, create_info.tablespace, strlen(create_info.tablespace), false)) { my_error(ER_OUT_OF_RESOURCES, MYF(ME_FATALERROR)); goto end_with_restore_list; } } // Reject invalid tablespace names specified for partitions. if (check_partition_tablespace_names(thd->lex->part_info)) goto end_with_restore_list; /* Fix names if symlinked or relocated tables */ if (append_file_to_dir(thd, &create_info.data_file_name, create_table->table_name) || append_file_to_dir(thd, &create_info.index_file_name, create_table->table_name)) goto end_with_restore_list; /* If no engine type was given, work out the default now rather than at parse-time. */ if (!(create_info.used_fields & HA_CREATE_USED_ENGINE)) create_info.db_type= create_info.options & HA_LEX_CREATE_TMP_TABLE ? ha_default_temp_handlerton(thd) : ha_default_handlerton(thd); /* If we are using SET CHARSET without DEFAULT, add an implicit DEFAULT to not confuse old users. (This may change). */ if ((create_info.used_fields & (HA_CREATE_USED_DEFAULT_CHARSET | HA_CREATE_USED_CHARSET)) == HA_CREATE_USED_CHARSET) { create_info.used_fields&= ~HA_CREATE_USED_CHARSET; create_info.used_fields|= HA_CREATE_USED_DEFAULT_CHARSET; create_info.default_table_charset= create_info.table_charset; create_info.table_charset= 0; } { partition_info *part_info= thd->lex->part_info; if (part_info != NULL && has_external_data_or_index_dir(*part_info) && check_access(thd, FILE_ACL, any_db, NULL, NULL, FALSE, FALSE)) { res= -1; goto end_with_restore_list; } if (part_info && !(part_info= thd->lex->part_info->get_clone(true))) { res= -1; goto end_with_restore_list; } thd->work_part_info= part_info; } if (select_lex->item_list.elements) // With select { Query_result *result; /* CREATE TABLE...IGNORE/REPLACE SELECT... can be unsafe, unless ORDER BY PRIMARY KEY clause is used in SELECT statement. We therefore use row based logging if mixed or row based logging is available. TODO: Check if the order of the output of the select statement is deterministic. Waiting for BUG#42415 */ if(lex->is_ignore()) lex->set_stmt_unsafe(LEX::BINLOG_STMT_UNSAFE_CREATE_IGNORE_SELECT); if(lex->duplicates == DUP_REPLACE) lex->set_stmt_unsafe(LEX::BINLOG_STMT_UNSAFE_CREATE_REPLACE_SELECT); /* If: a) we inside an SP and there was NAME_CONST substitution, b) binlogging is on (STMT mode), c) we log the SP as separate statements raise a warning, as it may cause problems (see 'NAME_CONST issues' in 'Binary Logging of Stored Programs') */ if (thd->query_name_consts && mysql_bin_log.is_open() && thd->variables.binlog_format == BINLOG_FORMAT_STMT && !mysql_bin_log.is_query_in_union(thd, thd->query_id)) { List_iterator_fast<Item> it(select_lex->item_list); Item *item; uint splocal_refs= 0; /* Count SP local vars in the top-level SELECT list */ while ((item= it++)) { if (item->is_splocal()) splocal_refs++; } /* If it differs from number of NAME_CONST substitution applied, we may have a SOME_FUNC(NAME_CONST()) in the SELECT list, that may cause a problem with binary log (see BUG#35383), raise a warning. */ if (splocal_refs != thd->query_name_consts) push_warning(thd, Sql_condition::SL_WARNING, ER_UNKNOWN_ERROR, "Invoked routine ran a statement that may cause problems with " "binary log, see 'NAME_CONST issues' in 'Binary Logging of Stored Programs' " "section of the manual."); } unit->set_limit(select_lex); /* Disable non-empty MERGE tables with CREATE...SELECT. Too complicated. See Bug #26379. Empty MERGE tables are read-only and don't allow CREATE...SELECT anyway. */ if (create_info.used_fields & HA_CREATE_USED_UNION) { my_error(ER_WRONG_OBJECT, MYF(0), create_table->db, create_table->table_name, "BASE TABLE"); res= 1; goto end_with_restore_list; } if (!(res= open_tables_for_query(thd, all_tables, 0))) { /* The table already exists */ if (create_table->table || create_table->is_view()) { if (create_info.options & HA_LEX_CREATE_IF_NOT_EXISTS) { push_warning_printf(thd, Sql_condition::SL_NOTE, ER_TABLE_EXISTS_ERROR, ER(ER_TABLE_EXISTS_ERROR), create_info.alias); my_ok(thd); } else { my_error(ER_TABLE_EXISTS_ERROR, MYF(0), create_info.alias); res= 1; } goto end_with_restore_list; } /* Remove target table from main select and name resolution context. This can't be done earlier as it will break view merging in statements like "CREATE TABLE IF NOT EXISTS existing_view SELECT". */ lex->unlink_first_table(&link_to_local); /* Updating any other table is prohibited in CTS statement */ for (TABLE_LIST *table= lex->query_tables; table; table= table->next_global) if (table->lock_type >= TL_WRITE_ALLOW_WRITE) { lex->link_first_table_back(create_table, link_to_local); res= 1; my_error(ER_CANT_UPDATE_TABLE_IN_CREATE_TABLE_SELECT, MYF(0), table->table_name, create_info.alias); goto end_with_restore_list; } /* Query_result_create is currently not re-execution friendly and needs to be created for every execution of a PS/SP. */ if ((result= new Query_result_create(create_table, &create_info, &alter_info, select_lex->item_list, lex->duplicates, select_tables))) { Ignore_error_handler ignore_handler; Strict_error_handler strict_handler; if (thd->lex->is_ignore()) thd->push_internal_handler(&ignore_handler); else if (thd->is_strict_mode()) thd->push_internal_handler(&strict_handler); /* CREATE from SELECT give its SELECT_LEX for SELECT, and item_list belong to SELECT */ res= handle_query(thd, lex, result, SELECT_NO_UNLOCK, 0); if (thd->lex->is_ignore() || thd->is_strict_mode()) thd->pop_internal_handler(); delete result; } lex->link_first_table_back(create_table, link_to_local); } } else { Strict_error_handler strict_handler; /* Push Strict_error_handler */ if (!thd->lex->is_ignore() && thd->is_strict_mode()) thd->push_internal_handler(&strict_handler); Partition_in_shared_ts_error_handler partition_in_shared_ts_handler; thd->push_internal_handler(&partition_in_shared_ts_handler); /* regular create */ if (create_info.options & HA_LEX_CREATE_TABLE_LIKE) { /* CREATE TABLE ... LIKE ... */ res= mysql_create_like_table(thd, create_table, select_tables, &create_info); } else { /* Regular CREATE TABLE */ res= mysql_create_table(thd, create_table, &create_info, &alter_info); } thd->pop_internal_handler(); /* Pop Strict_error_handler */ if (!thd->lex->is_ignore() && thd->is_strict_mode()) thd->pop_internal_handler(); if (!res) { /* in case of create temp tables if @@session_track_state_change is ON then send session state notification in OK packet */ if(create_info.options & HA_LEX_CREATE_TMP_TABLE && thd->session_tracker.get_tracker(SESSION_STATE_CHANGE_TRACKER)->is_enabled()) thd->session_tracker.get_tracker(SESSION_STATE_CHANGE_TRACKER)->mark_as_changed(thd, NULL); my_ok(thd); } } end_with_restore_list: break; } case SQLCOM_CREATE_INDEX: /* Fall through */ case SQLCOM_DROP_INDEX: /* CREATE INDEX and DROP INDEX are implemented by calling ALTER TABLE with proper arguments. In the future ALTER TABLE will notice that the request is to only add indexes and create these one by one for the existing table without having to do a full rebuild. */ { /* Prepare stack copies to be re-execution safe */ HA_CREATE_INFO create_info; Alter_info alter_info(lex->alter_info, thd->mem_root); if (thd->is_fatal_error) /* out of memory creating a copy of alter_info */ goto error; assert(first_table == all_tables && first_table != 0); if (check_one_table_access(thd, INDEX_ACL, all_tables)) goto error; /* purecov: inspected */ /* Currently CREATE INDEX or DROP INDEX cause a full table rebuild and thus classify as slow administrative statements just like ALTER TABLE. */ thd->enable_slow_log= opt_log_slow_admin_statements; create_info.db_type= 0; create_info.row_type= ROW_TYPE_NOT_USED; create_info.default_table_charset= thd->variables.collation_database; /* Push Strict_error_handler */ Strict_error_handler strict_handler; if (thd->is_strict_mode()) thd->push_internal_handler(&strict_handler); assert(!select_lex->order_list.elements); res= mysql_alter_table(thd, first_table->db, first_table->table_name, &create_info, first_table, &alter_info); /* Pop Strict_error_handler */ if (thd->is_strict_mode()) thd->pop_internal_handler(); break; } #ifdef HAVE_REPLICATION case SQLCOM_START_GROUP_REPLICATION: { if (check_global_access(thd, SUPER_ACL)) goto error; /* If the client thread has locked tables, a deadlock is possible. Assume that - the client thread does LOCK TABLE t READ. - then the client thread does START GROUP_REPLICATION. -try to make the server in super ready only mode -acquire MDL lock ownership which will be waiting for LOCK on table t to be released. To prevent that, refuse START GROUP_REPLICATION if the client thread has locked tables */ if (thd->locked_tables_mode || thd->in_active_multi_stmt_transaction() || thd->in_sub_stmt) { my_message(ER_LOCK_OR_ACTIVE_TRANSACTION, ER(ER_LOCK_OR_ACTIVE_TRANSACTION), MYF(0)); goto error; } res= group_replication_start(); //To reduce server dependency, server errors are not used here switch (res) { case 1: //GROUP_REPLICATION_CONFIGURATION_ERROR my_message(ER_GROUP_REPLICATION_CONFIGURATION, ER(ER_GROUP_REPLICATION_CONFIGURATION), MYF(0)); goto error; case 2: //GROUP_REPLICATION_ALREADY_RUNNING my_message(ER_GROUP_REPLICATION_RUNNING, ER(ER_GROUP_REPLICATION_RUNNING), MYF(0)); goto error; case 3: //GROUP_REPLICATION_REPLICATION_APPLIER_INIT_ERROR my_message(ER_GROUP_REPLICATION_APPLIER_INIT_ERROR, ER(ER_GROUP_REPLICATION_APPLIER_INIT_ERROR), MYF(0)); goto error; case 4: //GROUP_REPLICATION_COMMUNICATION_LAYER_SESSION_ERROR my_message(ER_GROUP_REPLICATION_COMMUNICATION_LAYER_SESSION_ERROR, ER(ER_GROUP_REPLICATION_COMMUNICATION_LAYER_SESSION_ERROR), MYF(0)); goto error; case 5: //GROUP_REPLICATION_COMMUNICATION_LAYER_JOIN_ERROR my_message(ER_GROUP_REPLICATION_COMMUNICATION_LAYER_JOIN_ERROR, ER(ER_GROUP_REPLICATION_COMMUNICATION_LAYER_JOIN_ERROR), MYF(0)); goto error; case 7: //GROUP_REPLICATION_MAX_GROUP_SIZE my_message(ER_GROUP_REPLICATION_MAX_GROUP_SIZE, ER(ER_GROUP_REPLICATION_MAX_GROUP_SIZE), MYF(0)); goto error; } my_ok(thd); res= 0; break; } case SQLCOM_STOP_GROUP_REPLICATION: { if (check_global_access(thd, SUPER_ACL)) goto error; /* Please see explanation @SQLCOM_SLAVE_STOP case to know the reason for thd->locked_tables_mode in the below if condition. */ if (thd->locked_tables_mode || thd->in_active_multi_stmt_transaction() || thd->in_sub_stmt) { my_message(ER_LOCK_OR_ACTIVE_TRANSACTION, ER(ER_LOCK_OR_ACTIVE_TRANSACTION), MYF(0)); goto error; } res= group_replication_stop(); if (res == 1) //GROUP_REPLICATION_CONFIGURATION_ERROR { my_message(ER_GROUP_REPLICATION_CONFIGURATION, ER(ER_GROUP_REPLICATION_CONFIGURATION), MYF(0)); goto error; } if (res == 6) //GROUP_REPLICATION_APPLIER_THREAD_TIMEOUT { my_message(ER_GROUP_REPLICATION_STOP_APPLIER_THREAD_TIMEOUT, ER(ER_GROUP_REPLICATION_STOP_APPLIER_THREAD_TIMEOUT), MYF(0)); goto error; } my_ok(thd); res= 0; break; } case SQLCOM_SLAVE_START: { res= start_slave_cmd(thd); break; } case SQLCOM_SLAVE_STOP: { /* If the client thread has locked tables, a deadlock is possible. Assume that - the client thread does LOCK TABLE t READ. - then the master updates t. - then the SQL slave thread wants to update t, so it waits for the client thread because t is locked by it. - then the client thread does SLAVE STOP. SLAVE STOP waits for the SQL slave thread to terminate its update t, which waits for the client thread because t is locked by it. To prevent that, refuse SLAVE STOP if the client thread has locked tables */ if (thd->locked_tables_mode || thd->in_active_multi_stmt_transaction() || thd->global_read_lock.is_acquired()) { my_message(ER_LOCK_OR_ACTIVE_TRANSACTION, ER(ER_LOCK_OR_ACTIVE_TRANSACTION), MYF(0)); goto error; } res= stop_slave_cmd(thd); break; } #endif /* HAVE_REPLICATION */ case SQLCOM_RENAME_TABLE: { assert(first_table == all_tables && first_table != 0); TABLE_LIST *table; for (table= first_table; table; table= table->next_local->next_local) { if (check_access(thd, ALTER_ACL | DROP_ACL, table->db, &table->grant.privilege, &table->grant.m_internal, 0, 0) || check_access(thd, INSERT_ACL | CREATE_ACL, table->next_local->db, &table->next_local->grant.privilege, &table->next_local->grant.m_internal, 0, 0)) goto error; TABLE_LIST old_list, new_list; /* we do not need initialize old_list and new_list because we will come table[0] and table->next[0] there */ old_list= table[0]; new_list= table->next_local[0]; if (check_grant(thd, ALTER_ACL | DROP_ACL, &old_list, FALSE, 1, FALSE) || (!test_all_bits(table->next_local->grant.privilege, INSERT_ACL | CREATE_ACL) && check_grant(thd, INSERT_ACL | CREATE_ACL, &new_list, FALSE, 1, FALSE))) goto error; } if (mysql_rename_tables(thd, first_table, 0)) goto error; break; } #ifndef EMBEDDED_LIBRARY case SQLCOM_SHOW_BINLOGS: { if (check_global_access(thd, SUPER_ACL | REPL_CLIENT_ACL)) goto error; res = show_binlogs(thd); break; } #endif /* EMBEDDED_LIBRARY */ case SQLCOM_SHOW_CREATE: assert(first_table == all_tables && first_table != 0); { /* Access check: SHOW CREATE TABLE require any privileges on the table level (ie effecting all columns in the table). SHOW CREATE VIEW require the SHOW_VIEW and SELECT ACLs on the table level. NOTE: SHOW_VIEW ACL is checked when the view is created. */ DBUG_PRINT("debug", ("lex->only_view: %d, table: %s.%s", lex->only_view, first_table->db, first_table->table_name)); if (lex->only_view) { if (check_table_access(thd, SELECT_ACL, first_table, FALSE, 1, FALSE)) { DBUG_PRINT("debug", ("check_table_access failed")); my_error(ER_TABLEACCESS_DENIED_ERROR, MYF(0), "SHOW", thd->security_context()->priv_user().str, thd->security_context()->host_or_ip().str, first_table->alias); goto error; } DBUG_PRINT("debug", ("check_table_access succeeded")); /* Ignore temporary tables if this is "SHOW CREATE VIEW" */ first_table->open_type= OT_BASE_ONLY; } else { /* Temporary tables should be opened for SHOW CREATE TABLE, but not for SHOW CREATE VIEW. */ if (open_temporary_tables(thd, all_tables)) goto error; /* The fact that check_some_access() returned FALSE does not mean that access is granted. We need to check if first_table->grant.privilege contains any table-specific privilege. */ DBUG_PRINT("debug", ("first_table->grant.privilege: %lx", first_table->grant.privilege)); if (check_some_access(thd, TABLE_OP_ACLS, first_table) || (first_table->grant.privilege & TABLE_OP_ACLS) == 0) { my_error(ER_TABLEACCESS_DENIED_ERROR, MYF(0), "SHOW", thd->security_context()->priv_user().str, thd->security_context()->host_or_ip().str, first_table->alias); goto error; } } /* Access is granted. Execute the command. */ res= mysqld_show_create(thd, first_table); break; } case SQLCOM_CHECKSUM: { assert(first_table == all_tables && first_table != 0); if (check_table_access(thd, SELECT_ACL, all_tables, FALSE, UINT_MAX, FALSE)) goto error; /* purecov: inspected */ res = mysql_checksum_table(thd, first_table, &lex->check_opt); break; } case SQLCOM_REPLACE: case SQLCOM_INSERT: case SQLCOM_REPLACE_SELECT: case SQLCOM_INSERT_SELECT: { assert(first_table == all_tables && first_table != 0); assert(lex->m_sql_cmd != NULL); res= lex->m_sql_cmd->execute(thd); break; } case SQLCOM_DELETE: case SQLCOM_DELETE_MULTI: case SQLCOM_UPDATE: case SQLCOM_UPDATE_MULTI: { assert(first_table == all_tables && first_table != 0); assert(lex->m_sql_cmd != NULL); res= lex->m_sql_cmd->execute(thd); break; } case SQLCOM_DROP_TABLE: { assert(first_table == all_tables && first_table != 0); if (!lex->drop_temporary) { if (check_table_access(thd, DROP_ACL, all_tables, FALSE, UINT_MAX, FALSE)) goto error; /* purecov: inspected */ } /* DDL and binlog write order are protected by metadata locks. */ res= mysql_rm_table(thd, first_table, lex->drop_if_exists, lex->drop_temporary); /* when dropping temporary tables if @@session_track_state_change is ON then send the boolean tracker in the OK packet */ if(!res && lex->drop_temporary) { if (thd->session_tracker.get_tracker(SESSION_STATE_CHANGE_TRACKER)->is_enabled()) thd->session_tracker.get_tracker(SESSION_STATE_CHANGE_TRACKER)->mark_as_changed(thd, NULL); } } break; case SQLCOM_SHOW_PRIVILEGES: res= mysqld_show_privileges(thd); break; case SQLCOM_SHOW_ENGINE_LOGS: { if (check_access(thd, FILE_ACL, any_db, NULL, NULL, 0, 0)) goto error; res= ha_show_status(thd, lex->create_info.db_type, HA_ENGINE_LOGS); break; } case SQLCOM_CHANGE_DB: { const LEX_CSTRING db_str= { select_lex->db, strlen(select_lex->db) }; if (!mysql_change_db(thd, db_str, FALSE)) my_ok(thd); break; } case SQLCOM_LOAD: { assert(first_table == all_tables && first_table != 0); uint privilege= (lex->duplicates == DUP_REPLACE ? INSERT_ACL | DELETE_ACL : INSERT_ACL) | (lex->local_file ? 0 : FILE_ACL); if (lex->local_file) { if (!thd->get_protocol()->has_client_capability(CLIENT_LOCAL_FILES) || !opt_local_infile) { my_message(ER_NOT_ALLOWED_COMMAND, ER(ER_NOT_ALLOWED_COMMAND), MYF(0)); goto error; } } if (check_one_table_access(thd, privilege, all_tables)) goto error; /* Push strict / ignore error handler */ Ignore_error_handler ignore_handler; Strict_error_handler strict_handler; if (thd->lex->is_ignore()) thd->push_internal_handler(&ignore_handler); else if (thd->is_strict_mode()) thd->push_internal_handler(&strict_handler); res= mysql_load(thd, lex->exchange, first_table, lex->load_field_list, lex->load_update_list, lex->load_value_list, lex->duplicates, lex->local_file); /* Pop ignore / strict error handler */ if (thd->lex->is_ignore() || thd->is_strict_mode()) thd->pop_internal_handler(); break; } case SQLCOM_SET_OPTION: { List<set_var_base> *lex_var_list= &lex->var_list; if ((check_table_access(thd, SELECT_ACL, all_tables, FALSE, UINT_MAX, FALSE) || open_and_lock_tables(thd, all_tables, 0))) goto error; if (!(res= sql_set_variables(thd, lex_var_list))) my_ok(thd); else { /* We encountered some sort of error, but no message was sent. Send something semi-generic here since we don't know which assignment in the list caused the error. */ if (!thd->is_error()) my_error(ER_WRONG_ARGUMENTS,MYF(0),"SET"); goto error; } break; } case SQLCOM_UNLOCK_TABLES: /* It is critical for mysqldump --single-transaction --master-data that UNLOCK TABLES does not implicitely commit a connection which has only done FLUSH TABLES WITH READ LOCK + BEGIN. If this assumption becomes false, mysqldump will not work. */ if (thd->variables.option_bits & OPTION_TABLE_LOCK) { /* Can we commit safely? If not, return to avoid releasing transactional metadata locks. */ if (trans_check_state(thd)) DBUG_RETURN(-1); res= trans_commit_implicit(thd); thd->locked_tables_list.unlock_locked_tables(thd); thd->mdl_context.release_transactional_locks(); thd->variables.option_bits&= ~(OPTION_TABLE_LOCK); } if (thd->global_read_lock.is_acquired()) thd->global_read_lock.unlock_global_read_lock(thd); if (res) goto error; my_ok(thd); break; case SQLCOM_LOCK_TABLES: /* Can we commit safely? If not, return to avoid releasing transactional metadata locks. */ if (trans_check_state(thd)) DBUG_RETURN(-1); /* We must end the transaction first, regardless of anything */ res= trans_commit_implicit(thd); thd->locked_tables_list.unlock_locked_tables(thd); /* Release transactional metadata locks. */ thd->mdl_context.release_transactional_locks(); if (res) goto error; /* Here we have to pre-open temporary tables for LOCK TABLES. CF_PREOPEN_TMP_TABLES is not set for this SQL statement simply because LOCK TABLES calls close_thread_tables() as a first thing (it's called from unlock_locked_tables() above). So even if CF_PREOPEN_TMP_TABLES was set and the tables would be pre-opened in a usual way, they would have been closed. */ if (open_temporary_tables(thd, all_tables)) goto error; if (lock_tables_precheck(thd, all_tables)) goto error; thd->variables.option_bits|= OPTION_TABLE_LOCK; res= lock_tables_open_and_lock_tables(thd, all_tables); if (res) { thd->variables.option_bits&= ~(OPTION_TABLE_LOCK); } else { if (thd->variables.query_cache_wlock_invalidate) query_cache.invalidate_locked_for_write(first_table); my_ok(thd); } break; case SQLCOM_CREATE_DB: { /* As mysql_create_db() may modify HA_CREATE_INFO structure passed to it, we need to use a copy of LEX::create_info to make execution prepared statement- safe. */ HA_CREATE_INFO create_info(lex->create_info); char *alias; if (!(alias=thd->strmake(lex->name.str, lex->name.length)) || (check_and_convert_db_name(&lex->name, FALSE) != IDENT_NAME_OK)) break; if (check_access(thd, CREATE_ACL, lex->name.str, NULL, NULL, 1, 0)) break; res= mysql_create_db(thd,(lower_case_table_names == 2 ? alias : lex->name.str), &create_info, 0); break; } case SQLCOM_DROP_DB: { if (check_and_convert_db_name(&lex->name, FALSE) != IDENT_NAME_OK) break; if (check_access(thd, DROP_ACL, lex->name.str, NULL, NULL, 1, 0)) break; res= mysql_rm_db(thd, to_lex_cstring(lex->name), lex->drop_if_exists, 0); break; } case SQLCOM_ALTER_DB_UPGRADE: { LEX_STRING *db= & lex->name; if (check_and_convert_db_name(db, FALSE) != IDENT_NAME_OK) break; if (check_access(thd, ALTER_ACL, db->str, NULL, NULL, 1, 0) || check_access(thd, DROP_ACL, db->str, NULL, NULL, 1, 0) || check_access(thd, CREATE_ACL, db->str, NULL, NULL, 1, 0)) { res= 1; break; } LEX_CSTRING db_name= {db->str, db->length}; res= mysql_upgrade_db(thd, db_name); if (!res) my_ok(thd); break; } case SQLCOM_ALTER_DB: { LEX_STRING *db= &lex->name; HA_CREATE_INFO create_info(lex->create_info); if (check_and_convert_db_name(db, FALSE) != IDENT_NAME_OK) break; if (check_access(thd, ALTER_ACL, db->str, NULL, NULL, 1, 0)) break; res= mysql_alter_db(thd, db->str, &create_info); break; } case SQLCOM_SHOW_CREATE_DB: { DBUG_EXECUTE_IF("4x_server_emul", my_error(ER_UNKNOWN_ERROR, MYF(0)); goto error;); if (check_and_convert_db_name(&lex->name, TRUE) != IDENT_NAME_OK) break; res= mysqld_show_create_db(thd, lex->name.str, &lex->create_info); break; } case SQLCOM_CREATE_EVENT: case SQLCOM_ALTER_EVENT: #ifndef EMBEDDED_LIBRARY do { assert(lex->event_parse_data); if (lex->table_or_sp_used()) { my_error(ER_NOT_SUPPORTED_YET, MYF(0), "Usage of subqueries or stored " "function calls as part of this statement"); break; } res= sp_process_definer(thd); if (res) break; switch (lex->sql_command) { case SQLCOM_CREATE_EVENT: { bool if_not_exists= (lex->create_info.options & HA_LEX_CREATE_IF_NOT_EXISTS); res= Events::create_event(thd, lex->event_parse_data, if_not_exists); break; } case SQLCOM_ALTER_EVENT: { LEX_STRING db_lex_str= NULL_STR; if (lex->spname) { db_lex_str.str= const_cast<char*>(lex->spname->m_db.str); db_lex_str.length= lex->spname->m_db.length; } res= Events::update_event(thd, lex->event_parse_data, lex->spname ? &db_lex_str : NULL, lex->spname ? &lex->spname->m_name : NULL); break; } default: assert(0); } DBUG_PRINT("info",("DDL error code=%d", res)); if (!res) my_ok(thd); } while (0); /* Don't do it, if we are inside a SP */ if (!thd->sp_runtime_ctx) { delete lex->sphead; lex->sphead= NULL; } /* lex->unit->cleanup() is called outside, no need to call it here */ break; case SQLCOM_SHOW_CREATE_EVENT: { LEX_STRING db_lex_str= {const_cast<char*>(lex->spname->m_db.str), lex->spname->m_db.length}; res= Events::show_create_event(thd, db_lex_str, lex->spname->m_name); break; } case SQLCOM_DROP_EVENT: { LEX_STRING db_lex_str= {const_cast<char*>(lex->spname->m_db.str), lex->spname->m_db.length}; if (!(res= Events::drop_event(thd, db_lex_str, lex->spname->m_name, lex->drop_if_exists))) my_ok(thd); break; } #else my_error(ER_NOT_SUPPORTED_YET,MYF(0),"embedded server"); break; #endif case SQLCOM_CREATE_FUNCTION: // UDF function { if (check_access(thd, INSERT_ACL, "mysql", NULL, NULL, 1, 0)) break; #ifdef HAVE_DLOPEN if (!(res = mysql_create_function(thd, &lex->udf))) my_ok(thd); #else my_error(ER_CANT_OPEN_LIBRARY, MYF(0), lex->udf.dl, 0, "feature disabled"); res= TRUE; #endif break; } #ifndef NO_EMBEDDED_ACCESS_CHECKS case SQLCOM_CREATE_USER: { if (check_access(thd, INSERT_ACL, "mysql", NULL, NULL, 1, 1) && check_global_access(thd,CREATE_USER_ACL)) break; /* Conditionally writes to binlog */ HA_CREATE_INFO create_info(lex->create_info); if (!(res = mysql_create_user(thd, lex->users_list, create_info.options & HA_LEX_CREATE_IF_NOT_EXISTS))) my_ok(thd); break; } case SQLCOM_DROP_USER: { if (check_access(thd, DELETE_ACL, "mysql", NULL, NULL, 1, 1) && check_global_access(thd,CREATE_USER_ACL)) break; /* Conditionally writes to binlog */ if (!(res = mysql_drop_user(thd, lex->users_list, lex->drop_if_exists))) my_ok(thd); break; } case SQLCOM_RENAME_USER: { if (check_access(thd, UPDATE_ACL, "mysql", NULL, NULL, 1, 1) && check_global_access(thd,CREATE_USER_ACL)) break; /* Conditionally writes to binlog */ if (!(res= mysql_rename_user(thd, lex->users_list))) my_ok(thd); break; } case SQLCOM_REVOKE_ALL: { if (check_access(thd, UPDATE_ACL, "mysql", NULL, NULL, 1, 1) && check_global_access(thd,CREATE_USER_ACL)) break; /* Replicate current user as grantor */ thd->binlog_invoker(); /* Conditionally writes to binlog */ if (!(res = mysql_revoke_all(thd, lex->users_list))) my_ok(thd); break; } case SQLCOM_REVOKE: case SQLCOM_GRANT: { if (lex->type != TYPE_ENUM_PROXY && check_access(thd, lex->grant | lex->grant_tot_col | GRANT_ACL, first_table ? first_table->db : select_lex->db, first_table ? &first_table->grant.privilege : NULL, first_table ? &first_table->grant.m_internal : NULL, first_table ? 0 : 1, 0)) goto error; /* Replicate current user as grantor */ thd->binlog_invoker(); if (thd->security_context()->user().str) // If not replication { LEX_USER *user, *tmp_user; bool first_user= TRUE; List_iterator <LEX_USER> user_list(lex->users_list); while ((tmp_user= user_list++)) { if (!(user= get_current_user(thd, tmp_user))) goto error; if (specialflag & SPECIAL_NO_RESOLVE && hostname_requires_resolving(user->host.str)) push_warning(thd, Sql_condition::SL_WARNING, ER_WARN_HOSTNAME_WONT_WORK, ER(ER_WARN_HOSTNAME_WONT_WORK)); // Are we trying to change a password of another user assert(user->host.str != 0); /* GRANT/REVOKE PROXY has the target user as a first entry in the list. */ if (lex->type == TYPE_ENUM_PROXY && first_user) { first_user= FALSE; if (acl_check_proxy_grant_access (thd, user->host.str, user->user.str, lex->grant & GRANT_ACL)) goto error; } else if (is_acl_user(user->host.str, user->user.str) && user->auth.str && check_change_password (thd, user->host.str, user->user.str, user->auth.str, user->auth.length)) goto error; } } if (first_table) { if (lex->type == TYPE_ENUM_PROCEDURE || lex->type == TYPE_ENUM_FUNCTION) { uint grants = lex->all_privileges ? (PROC_OP_ACLS) | (lex->grant & GRANT_ACL) : lex->grant; if (check_grant_routine(thd, grants | GRANT_ACL, all_tables, lex->type == TYPE_ENUM_PROCEDURE, 0)) goto error; /* Conditionally writes to binlog */ res= mysql_routine_grant(thd, all_tables, lex->type == TYPE_ENUM_PROCEDURE, lex->users_list, grants, lex->sql_command == SQLCOM_REVOKE, TRUE); if (!res) my_ok(thd); } else { if (check_grant(thd,(lex->grant | lex->grant_tot_col | GRANT_ACL), all_tables, FALSE, UINT_MAX, FALSE)) goto error; /* Conditionally writes to binlog */ res= mysql_table_grant(thd, all_tables, lex->users_list, lex->columns, lex->grant, lex->sql_command == SQLCOM_REVOKE); } } else { if (lex->columns.elements || (lex->type && lex->type != TYPE_ENUM_PROXY)) { my_message(ER_ILLEGAL_GRANT_FOR_TABLE, ER(ER_ILLEGAL_GRANT_FOR_TABLE), MYF(0)); goto error; } else { /* Conditionally writes to binlog */ res = mysql_grant(thd, select_lex->db, lex->users_list, lex->grant, lex->sql_command == SQLCOM_REVOKE, lex->type == TYPE_ENUM_PROXY); } if (!res) { if (lex->sql_command == SQLCOM_GRANT) { List_iterator <LEX_USER> str_list(lex->users_list); LEX_USER *user, *tmp_user; while ((tmp_user=str_list++)) { if (!(user= get_current_user(thd, tmp_user))) goto error; reset_mqh(user, 0); } } } } break; } #endif /*!NO_EMBEDDED_ACCESS_CHECKS*/ case SQLCOM_RESET: /* RESET commands are never written to the binary log, so we have to initialize this variable because RESET shares the same code as FLUSH */ lex->no_write_to_binlog= 1; // Fall through. case SQLCOM_FLUSH: { int write_to_binlog; if (check_global_access(thd,RELOAD_ACL)) goto error; if (first_table && lex->type & REFRESH_READ_LOCK) { /* Check table-level privileges. */ if (check_table_access(thd, LOCK_TABLES_ACL | SELECT_ACL, all_tables, FALSE, UINT_MAX, FALSE)) goto error; if (flush_tables_with_read_lock(thd, all_tables)) goto error; my_ok(thd); break; } else if (first_table && lex->type & REFRESH_FOR_EXPORT) { /* Check table-level privileges. */ if (check_table_access(thd, LOCK_TABLES_ACL | SELECT_ACL, all_tables, FALSE, UINT_MAX, FALSE)) goto error; if (flush_tables_for_export(thd, all_tables)) goto error; my_ok(thd); break; } /* reload_acl_and_cache() will tell us if we are allowed to write to the binlog or not. */ if (!reload_acl_and_cache(thd, lex->type, first_table, &write_to_binlog)) { /* We WANT to write and we CAN write. ! we write after unlocking the table. */ /* Presumably, RESET and binlog writing doesn't require synchronization */ if (write_to_binlog > 0) // we should write { if (!lex->no_write_to_binlog) res= write_bin_log(thd, false, thd->query().str, thd->query().length); } else if (write_to_binlog < 0) { /* We should not write, but rather report error because reload_acl_and_cache binlog interactions failed */ res= 1; } if (!res) my_ok(thd); } break; } case SQLCOM_KILL: { Item *it= lex->kill_value_list.head(); if (lex->table_or_sp_used()) { my_error(ER_NOT_SUPPORTED_YET, MYF(0), "Usage of subqueries or stored " "function calls as part of this statement"); goto error; } if ((!it->fixed && it->fix_fields(lex->thd, &it)) || it->check_cols(1)) { my_error(ER_SET_CONSTANTS_ONLY, MYF(0)); goto error; } my_thread_id thread_id= static_cast<my_thread_id>(it->val_int()); if (thd->is_error()) goto error; sql_kill(thd, thread_id, lex->type & ONLY_KILL_QUERY); break; } #ifndef NO_EMBEDDED_ACCESS_CHECKS case SQLCOM_SHOW_GRANTS: { LEX_USER *grant_user= get_current_user(thd, lex->grant_user); if (!grant_user) goto error; if (!strcmp(thd->security_context()->priv_user().str, grant_user->user.str) || !check_access(thd, SELECT_ACL, "mysql", NULL, NULL, 1, 0)) { res = mysql_show_grants(thd, grant_user); } break; } case SQLCOM_SHOW_CREATE_USER: { LEX_USER *show_user= get_current_user(thd, lex->grant_user); Security_context *sctx= thd->security_context(); bool are_both_users_same= !strcmp(sctx->priv_user().str, show_user->user.str) && !my_strcasecmp(system_charset_info, show_user->host.str, sctx->priv_host().str); if (are_both_users_same || !check_access(thd, SELECT_ACL, "mysql", NULL, NULL, 1, 0)) res= mysql_show_create_user(thd, show_user, are_both_users_same); break; } #endif case SQLCOM_BEGIN: if (trans_begin(thd, lex->start_transaction_opt)) goto error; my_ok(thd); break; case SQLCOM_COMMIT: { assert(thd->lock == NULL || thd->locked_tables_mode == LTM_LOCK_TABLES); bool tx_chain= (lex->tx_chain == TVL_YES || (thd->variables.completion_type == 1 && lex->tx_chain != TVL_NO)); bool tx_release= (lex->tx_release == TVL_YES || (thd->variables.completion_type == 2 && lex->tx_release != TVL_NO)); if (trans_commit(thd)) goto error; thd->mdl_context.release_transactional_locks(); /* Begin transaction with the same isolation level. */ if (tx_chain) { if (trans_begin(thd)) goto error; } else { /* Reset the isolation level and access mode if no chaining transaction.*/ trans_reset_one_shot_chistics(thd); } /* Disconnect the current client connection. */ if (tx_release) thd->killed= THD::KILL_CONNECTION; my_ok(thd); break; } case SQLCOM_ROLLBACK: { assert(thd->lock == NULL || thd->locked_tables_mode == LTM_LOCK_TABLES); bool tx_chain= (lex->tx_chain == TVL_YES || (thd->variables.completion_type == 1 && lex->tx_chain != TVL_NO)); bool tx_release= (lex->tx_release == TVL_YES || (thd->variables.completion_type == 2 && lex->tx_release != TVL_NO)); if (trans_rollback(thd)) goto error; thd->mdl_context.release_transactional_locks(); /* Begin transaction with the same isolation level. */ if (tx_chain) { if (trans_begin(thd)) goto error; } else { /* Reset the isolation level and access mode if no chaining transaction.*/ trans_reset_one_shot_chistics(thd); } /* Disconnect the current client connection. */ if (tx_release) thd->killed= THD::KILL_CONNECTION; my_ok(thd); break; } case SQLCOM_RELEASE_SAVEPOINT: if (trans_release_savepoint(thd, lex->ident)) goto error; my_ok(thd); break; case SQLCOM_ROLLBACK_TO_SAVEPOINT: if (trans_rollback_to_savepoint(thd, lex->ident)) goto error; my_ok(thd); break; case SQLCOM_SAVEPOINT: if (trans_savepoint(thd, lex->ident)) goto error; my_ok(thd); break; case SQLCOM_CREATE_PROCEDURE: case SQLCOM_CREATE_SPFUNCTION: { uint namelen; char *name; assert(lex->sphead != 0); assert(lex->sphead->m_db.str); /* Must be initialized in the parser */ /* Verify that the database name is allowed, optionally lowercase it. */ if (check_and_convert_db_name(&lex->sphead->m_db, FALSE) != IDENT_NAME_OK) goto error; if (check_access(thd, CREATE_PROC_ACL, lex->sphead->m_db.str, NULL, NULL, 0, 0)) goto error; name= lex->sphead->name(&namelen); #ifdef HAVE_DLOPEN if (lex->sphead->m_type == SP_TYPE_FUNCTION) { udf_func *udf = find_udf(name, namelen); if (udf) { my_error(ER_UDF_EXISTS, MYF(0), name); goto error; } } #endif if (sp_process_definer(thd)) goto error; /* Record the CURRENT_USER in binlog. The CURRENT_USER is used on slave to grant default privileges when sp_automatic_privileges variable is set. */ thd->binlog_invoker(); if (! (res= sp_create_routine(thd, lex->sphead))) { #ifndef NO_EMBEDDED_ACCESS_CHECKS /* only add privileges if really neccessary */ Security_context security_context; bool restore_backup_context= false; Security_context *backup= NULL; /* We're going to issue an implicit GRANT statement so we close all open tables. We have to keep metadata locks as this ensures that this statement is atomic against concurent FLUSH TABLES WITH READ LOCK. Deadlocks which can arise due to fact that this implicit statement takes metadata locks should be detected by a deadlock detector in MDL subsystem and reported as errors. No need to commit/rollback statement transaction, it's not started. TODO: Long-term we should either ensure that implicit GRANT statement is written into binary log as a separate statement or make both creation of routine and implicit GRANT parts of one fully atomic statement. */ assert(thd->get_transaction()->is_empty(Transaction_ctx::STMT)); close_thread_tables(thd); /* Check if invoker exists on slave, then use invoker privilege to insert routine privileges to mysql.procs_priv. If invoker is not available then consider using definer. Check if the definer exists on slave, then use definer privilege to insert routine privileges to mysql.procs_priv. For current user of SQL thread has GLOBAL_ACL privilege, which doesn't any check routine privileges, so no routine privilege record will insert into mysql.procs_priv. */ if (thd->slave_thread) { LEX_CSTRING current_user; LEX_CSTRING current_host; if (thd->has_invoker()) { current_host= thd->get_invoker_host(); current_user= thd->get_invoker_user(); } else { current_host= lex->definer->host; current_user= lex->definer->user; } if (is_acl_user(current_host.str, current_user.str)) { security_context.change_security_context(thd, current_user, current_host, &thd->lex->sphead->m_db, &backup); restore_backup_context= true; } } if (sp_automatic_privileges && !opt_noacl && check_routine_access(thd, DEFAULT_CREATE_PROC_ACLS, lex->sphead->m_db.str, name, lex->sql_command == SQLCOM_CREATE_PROCEDURE, 1)) { if (sp_grant_privileges(thd, lex->sphead->m_db.str, name, lex->sql_command == SQLCOM_CREATE_PROCEDURE)) push_warning(thd, Sql_condition::SL_WARNING, ER_PROC_AUTO_GRANT_FAIL, ER(ER_PROC_AUTO_GRANT_FAIL)); thd->clear_error(); } /* Restore current user with GLOBAL_ACL privilege of SQL thread */ if (restore_backup_context) { assert(thd->slave_thread == 1); thd->security_context()->restore_security_context(thd, backup); } #endif my_ok(thd); } break; /* break super switch */ } /* end case group bracket */ case SQLCOM_CALL: { sp_head *sp; /* Here we check for the execute privilege on stored procedure. */ if (check_routine_access(thd, EXECUTE_ACL, lex->spname->m_db.str, lex->spname->m_name.str, lex->sql_command == SQLCOM_CALL, 0)) goto error; /* This will cache all SP and SF and open and lock all tables required for execution. */ if (check_table_access(thd, SELECT_ACL, all_tables, FALSE, UINT_MAX, FALSE) || open_and_lock_tables(thd, all_tables, 0)) goto error; /* By this moment all needed SPs should be in cache so no need to look into DB. */ if (!(sp= sp_find_routine(thd, SP_TYPE_PROCEDURE, lex->spname, &thd->sp_proc_cache, TRUE))) { my_error(ER_SP_DOES_NOT_EXIST, MYF(0), "PROCEDURE", lex->spname->m_qname.str); goto error; } else { ha_rows select_limit; /* bits that should be cleared in thd->server_status */ uint bits_to_be_cleared= 0; /* Check that the stored procedure doesn't contain Dynamic SQL and doesn't return result sets: such stored procedures can't be called from a function or trigger. */ if (thd->in_sub_stmt) { const char *where= (thd->in_sub_stmt & SUB_STMT_TRIGGER ? "trigger" : "function"); if (sp->is_not_allowed_in_function(where)) goto error; } #ifndef EMBEDDED_LIBRARY if (mysql_audit_notify(thd, AUDIT_EVENT(MYSQL_AUDIT_STORED_PROGRAM_EXECUTE), lex->spname->m_db.str, lex->spname->m_name.str, NULL)) { goto error; } #endif /* !EMBEDDED_LIBRARY */ if (sp->m_flags & sp_head::MULTI_RESULTS) { if (!thd->get_protocol()->has_client_capability(CLIENT_MULTI_RESULTS)) { /* The client does not support multiple result sets being sent back */ my_error(ER_SP_BADSELECT, MYF(0), sp->m_qname.str); goto error; } /* If SERVER_MORE_RESULTS_EXISTS is not set, then remember that it should be cleared */ bits_to_be_cleared= (~thd->server_status & SERVER_MORE_RESULTS_EXISTS); thd->server_status|= SERVER_MORE_RESULTS_EXISTS; } if (check_routine_access(thd, EXECUTE_ACL, sp->m_db.str, sp->m_name.str, TRUE, FALSE)) { goto error; } select_limit= thd->variables.select_limit; thd->variables.select_limit= HA_POS_ERROR; /* We never write CALL statements into binlog: - If the mode is non-prelocked, each statement will be logged separately. - If the mode is prelocked, the invoking statement will care about writing into binlog. So just execute the statement. */ res= sp->execute_procedure(thd, &lex->call_value_list); thd->variables.select_limit= select_limit; thd->server_status&= ~bits_to_be_cleared; if (!res) { my_ok(thd, (thd->get_row_count_func() < 0) ? 0 : thd->get_row_count_func()); } else { assert(thd->is_error() || thd->killed); goto error; // Substatement should already have sent error } } break; } case SQLCOM_ALTER_PROCEDURE: case SQLCOM_ALTER_FUNCTION: { if (check_routine_access(thd, ALTER_PROC_ACL, lex->spname->m_db.str, lex->spname->m_name.str, lex->sql_command == SQLCOM_ALTER_PROCEDURE, false)) goto error; enum_sp_type sp_type= (lex->sql_command == SQLCOM_ALTER_PROCEDURE) ? SP_TYPE_PROCEDURE : SP_TYPE_FUNCTION; /* Note that if you implement the capability of ALTER FUNCTION to alter the body of the function, this command should be made to follow the restrictions that log-bin-trust-function-creators=0 already puts on CREATE FUNCTION. */ /* Conditionally writes to binlog */ int sp_result= sp_update_routine(thd, sp_type, lex->spname, &lex->sp_chistics); if (thd->killed) goto error; switch (sp_result) { case SP_OK: my_ok(thd); break; case SP_KEY_NOT_FOUND: my_error(ER_SP_DOES_NOT_EXIST, MYF(0), SP_COM_STRING(lex), lex->spname->m_qname.str); goto error; default: my_error(ER_SP_CANT_ALTER, MYF(0), SP_COM_STRING(lex), lex->spname->m_qname.str); goto error; } break; } case SQLCOM_DROP_PROCEDURE: case SQLCOM_DROP_FUNCTION: { #ifdef HAVE_DLOPEN if (lex->sql_command == SQLCOM_DROP_FUNCTION && ! lex->spname->m_explicit_name) { /* DROP FUNCTION <non qualified name> */ udf_func *udf = find_udf(lex->spname->m_name.str, lex->spname->m_name.length); if (udf) { if (check_access(thd, DELETE_ACL, "mysql", NULL, NULL, 1, 0)) goto error; if (!(res = mysql_drop_function(thd, &lex->spname->m_name))) { my_ok(thd); break; } my_error(ER_SP_DROP_FAILED, MYF(0), "FUNCTION (UDF)", lex->spname->m_name.str); goto error; } if (lex->spname->m_db.str == NULL) { if (lex->drop_if_exists) { push_warning_printf(thd, Sql_condition::SL_NOTE, ER_SP_DOES_NOT_EXIST, ER(ER_SP_DOES_NOT_EXIST), "FUNCTION (UDF)", lex->spname->m_name.str); res= FALSE; my_ok(thd); break; } my_error(ER_SP_DOES_NOT_EXIST, MYF(0), "FUNCTION (UDF)", lex->spname->m_name.str); goto error; } /* Fall thought to test for a stored function */ } #endif const char *db= lex->spname->m_db.str; char *name= lex->spname->m_name.str; if (check_routine_access(thd, ALTER_PROC_ACL, db, name, lex->sql_command == SQLCOM_DROP_PROCEDURE, false)) goto error; enum_sp_type sp_type= (lex->sql_command == SQLCOM_DROP_PROCEDURE) ? SP_TYPE_PROCEDURE : SP_TYPE_FUNCTION; /* Conditionally writes to binlog */ int sp_result= sp_drop_routine(thd, sp_type, lex->spname); #ifndef NO_EMBEDDED_ACCESS_CHECKS /* We're going to issue an implicit REVOKE statement so we close all open tables. We have to keep metadata locks as this ensures that this statement is atomic against concurent FLUSH TABLES WITH READ LOCK. Deadlocks which can arise due to fact that this implicit statement takes metadata locks should be detected by a deadlock detector in MDL subsystem and reported as errors. No need to commit/rollback statement transaction, it's not started. TODO: Long-term we should either ensure that implicit REVOKE statement is written into binary log as a separate statement or make both dropping of routine and implicit REVOKE parts of one fully atomic statement. */ assert(thd->get_transaction()->is_empty(Transaction_ctx::STMT)); close_thread_tables(thd); if (sp_result != SP_KEY_NOT_FOUND && sp_automatic_privileges && !opt_noacl && sp_revoke_privileges(thd, db, name, lex->sql_command == SQLCOM_DROP_PROCEDURE)) { push_warning(thd, Sql_condition::SL_WARNING, ER_PROC_AUTO_REVOKE_FAIL, ER(ER_PROC_AUTO_REVOKE_FAIL)); /* If this happens, an error should have been reported. */ goto error; } #endif res= sp_result; switch (sp_result) { case SP_OK: my_ok(thd); break; case SP_KEY_NOT_FOUND: if (lex->drop_if_exists) { res= write_bin_log(thd, true, thd->query().str, thd->query().length); push_warning_printf(thd, Sql_condition::SL_NOTE, ER_SP_DOES_NOT_EXIST, ER(ER_SP_DOES_NOT_EXIST), SP_COM_STRING(lex), lex->spname->m_qname.str); if (!res) my_ok(thd); break; } my_error(ER_SP_DOES_NOT_EXIST, MYF(0), SP_COM_STRING(lex), lex->spname->m_qname.str); goto error; default: my_error(ER_SP_DROP_FAILED, MYF(0), SP_COM_STRING(lex), lex->spname->m_qname.str); goto error; } break; } case SQLCOM_SHOW_CREATE_PROC: { if (sp_show_create_routine(thd, SP_TYPE_PROCEDURE, lex->spname)) goto error; break; } case SQLCOM_SHOW_CREATE_FUNC: { if (sp_show_create_routine(thd, SP_TYPE_FUNCTION, lex->spname)) goto error; break; } case SQLCOM_SHOW_PROC_CODE: case SQLCOM_SHOW_FUNC_CODE: { #ifndef NDEBUG sp_head *sp; enum_sp_type sp_type= (lex->sql_command == SQLCOM_SHOW_PROC_CODE) ? SP_TYPE_PROCEDURE : SP_TYPE_FUNCTION; if (sp_cache_routine(thd, sp_type, lex->spname, false, &sp)) goto error; if (!sp || sp->show_routine_code(thd)) { /* We don't distinguish between errors for now */ my_error(ER_SP_DOES_NOT_EXIST, MYF(0), SP_COM_STRING(lex), lex->spname->m_name.str); goto error; } break; #else my_error(ER_FEATURE_DISABLED, MYF(0), "SHOW PROCEDURE|FUNCTION CODE", "--with-debug"); goto error; #endif // ifndef NDEBUG } case SQLCOM_SHOW_CREATE_TRIGGER: { if (lex->spname->m_name.length > NAME_LEN) { my_error(ER_TOO_LONG_IDENT, MYF(0), lex->spname->m_name.str); goto error; } if (show_create_trigger(thd, lex->spname)) goto error; /* Error has been already logged. */ break; } case SQLCOM_CREATE_VIEW: { /* Note: SQLCOM_CREATE_VIEW also handles 'ALTER VIEW' commands as specified through the thd->lex->create_view_mode flag. */ res= mysql_create_view(thd, first_table, thd->lex->create_view_mode); break; } case SQLCOM_DROP_VIEW: { if (check_table_access(thd, DROP_ACL, all_tables, FALSE, UINT_MAX, FALSE)) goto error; /* Conditionally writes to binlog. */ res= mysql_drop_view(thd, first_table, thd->lex->drop_mode); break; } case SQLCOM_CREATE_TRIGGER: { /* Conditionally writes to binlog. */ res= mysql_create_or_drop_trigger(thd, all_tables, 1); break; } case SQLCOM_DROP_TRIGGER: { /* Conditionally writes to binlog. */ res= mysql_create_or_drop_trigger(thd, all_tables, 0); break; } case SQLCOM_ALTER_TABLESPACE: if (check_global_access(thd, CREATE_TABLESPACE_ACL)) break; if (!(res= mysql_alter_tablespace(thd, lex->alter_tablespace_info))) my_ok(thd); break; case SQLCOM_BINLOG_BASE64_EVENT: { #ifndef EMBEDDED_LIBRARY mysql_client_binlog_statement(thd); #else /* EMBEDDED_LIBRARY */ my_error(ER_OPTION_PREVENTS_STATEMENT, MYF(0), "embedded"); #endif /* EMBEDDED_LIBRARY */ break; } case SQLCOM_EXPLAIN_OTHER: { /* EXPLAIN FOR CONNECTION <id> */ mysql_explain_other(thd); break; } case SQLCOM_ANALYZE: case SQLCOM_CHECK: case SQLCOM_OPTIMIZE: case SQLCOM_REPAIR: case SQLCOM_TRUNCATE: case SQLCOM_ALTER_TABLE: case SQLCOM_HA_OPEN: case SQLCOM_HA_READ: case SQLCOM_HA_CLOSE: assert(first_table == all_tables && first_table != 0); /* fall through */ case SQLCOM_CREATE_SERVER: case SQLCOM_ALTER_SERVER: case SQLCOM_DROP_SERVER: case SQLCOM_SIGNAL: case SQLCOM_RESIGNAL: case SQLCOM_GET_DIAGNOSTICS: case SQLCOM_CHANGE_REPLICATION_FILTER: case SQLCOM_XA_START: case SQLCOM_XA_END: case SQLCOM_XA_PREPARE: case SQLCOM_XA_COMMIT: case SQLCOM_XA_ROLLBACK: case SQLCOM_XA_RECOVER: case SQLCOM_INSTALL_PLUGIN: case SQLCOM_UNINSTALL_PLUGIN: case SQLCOM_SHUTDOWN: case SQLCOM_ALTER_INSTANCE: case SQLCOM_SHOW_PROCESSLIST: assert(lex->m_sql_cmd != NULL); res= lex->m_sql_cmd->execute(thd); break; #ifndef NO_EMBEDDED_ACCESS_CHECKS case SQLCOM_ALTER_USER: { LEX_USER *user, *tmp_user; bool changing_own_password= false; bool own_password_expired= thd->security_context()->password_expired(); bool check_permission= true; List_iterator <LEX_USER> user_list(lex->users_list); while ((tmp_user= user_list++)) { bool update_password_only= FALSE; bool is_self= false; /* If it is an empty lex_user update it with current user */ if (!tmp_user->host.str && !tmp_user->user.str) { /* set user information as of the current user */ assert(thd->security_context()->priv_host().str); tmp_user->host.str= (char *) thd->security_context()->priv_host().str; tmp_user->host.length= strlen(thd->security_context()->priv_host().str); assert(thd->security_context()->user().str); tmp_user->user.str= (char *) thd->security_context()->user().str; tmp_user->user.length= strlen(thd->security_context()->user().str); } if (!(user= get_current_user(thd, tmp_user))) goto error; /* copy password expire attributes to individual lex user */ user->alter_status= thd->lex->alter_password; if (user->uses_identified_by_clause && !user->uses_identified_with_clause && !thd->lex->mqh.specified_limits && !user->alter_status.update_account_locked_column && !user->alter_status.update_password_expired_column && !user->alter_status.expire_after_days && user->alter_status.use_default_password_lifetime && (thd->lex->ssl_type == SSL_TYPE_NOT_SPECIFIED)) update_password_only= TRUE; is_self= !strcmp(thd->security_context()->user().length ? thd->security_context()->user().str : "", user->user.str) && !my_strcasecmp(&my_charset_latin1, user->host.str, thd->security_context()->priv_host().str); /* if user executes ALTER statement to change password only for himself then skip access check. */ if (update_password_only && is_self) { changing_own_password= true; continue; } else if (check_permission) { if (check_access(thd, UPDATE_ACL, "mysql", NULL, NULL, 1, 1) && check_global_access(thd, CREATE_USER_ACL)) goto error; check_permission= false; } if (is_self && (user->uses_identified_by_clause || user->uses_identified_with_clause || user->uses_authentication_string_clause || user->uses_identified_by_password_clause)) { changing_own_password= true; break; } if (update_password_only && likely((get_server_state() == SERVER_OPERATING)) && !strcmp(thd->security_context()->priv_user().str,"")) { my_message(ER_PASSWORD_ANONYMOUS_USER, ER(ER_PASSWORD_ANONYMOUS_USER), MYF(0)); goto error; } } if (unlikely(own_password_expired && !changing_own_password)) { my_error(ER_MUST_CHANGE_PASSWORD, MYF(0)); goto error; } /* Conditionally writes to binlog */ if (!(res = mysql_alter_user(thd, lex->users_list, lex->drop_if_exists))) my_ok(thd); break; } #endif default: #ifndef EMBEDDED_LIBRARY assert(0); /* Impossible */ #endif my_ok(thd); break; } goto finish; error: res= TRUE; finish: THD_STAGE_INFO(thd, stage_query_end); if (res && thd->get_reprepare_observer() != NULL && thd->get_reprepare_observer()->is_invalidated() && thd->get_reprepare_observer()->can_retry()) { thd->skip_gtid_rollback = true; } // Cleanup EXPLAIN info if (!thd->in_sub_stmt) { if (is_explainable_query(lex->sql_command)) { DEBUG_SYNC(thd, "before_reset_query_plan"); /* We want EXPLAIN CONNECTION to work until the explained statement ends, thus it is only now that we may fully clean up any unit of this statement. */ lex->unit->assert_not_fully_clean(); } thd->query_plan.set_query_plan(SQLCOM_END, NULL, false); } assert(!thd->in_active_multi_stmt_transaction() || thd->in_multi_stmt_transaction_mode()); if (! thd->in_sub_stmt) { #ifndef EMBEDDED_LIBRARY mysql_audit_notify(thd, first_level ? MYSQL_AUDIT_QUERY_STATUS_END : MYSQL_AUDIT_QUERY_NESTED_STATUS_END, first_level ? "MYSQL_AUDIT_QUERY_STATUS_END" : "MYSQL_AUDIT_QUERY_NESTED_STATUS_END"); #endif /* !EMBEDDED_LIBRARY */ /* report error issued during command execution */ if (thd->killed_errno()) thd->send_kill_message(); if (thd->is_error() || (thd->variables.option_bits & OPTION_MASTER_SQL_ERROR)) trans_rollback_stmt(thd); else { /* If commit fails, we should be able to reset the OK status. */ thd->get_stmt_da()->set_overwrite_status(true); trans_commit_stmt(thd); thd->get_stmt_da()->set_overwrite_status(false); } if (thd->killed == THD::KILL_QUERY || thd->killed == THD::KILL_TIMEOUT || thd->killed == THD::KILL_BAD_DATA) { thd->killed= THD::NOT_KILLED; thd->reset_query_for_display(); } } lex->unit->cleanup(true); /* Free tables */ THD_STAGE_INFO(thd, stage_closing_tables); close_thread_tables(thd); #ifndef NDEBUG if (lex->sql_command != SQLCOM_SET_OPTION && ! thd->in_sub_stmt) DEBUG_SYNC(thd, "execute_command_after_close_tables"); #endif if (! thd->in_sub_stmt && thd->transaction_rollback_request) { /* We are not in sub-statement and transaction rollback was requested by one of storage engines (e.g. due to deadlock). Rollback transaction in all storage engines including binary log. */ trans_rollback_implicit(thd); thd->mdl_context.release_transactional_locks(); } else if (stmt_causes_implicit_commit(thd, CF_IMPLICIT_COMMIT_END)) { /* No transaction control allowed in sub-statements. */ assert(! thd->in_sub_stmt); /* If commit fails, we should be able to reset the OK status. */ thd->get_stmt_da()->set_overwrite_status(true); /* Commit the normal transaction if one is active. */ trans_commit_implicit(thd); thd->get_stmt_da()->set_overwrite_status(false); thd->mdl_context.release_transactional_locks(); } else if (! thd->in_sub_stmt && ! thd->in_multi_stmt_transaction_mode()) { /* - If inside a multi-statement transaction, defer the release of metadata locks until the current transaction is either committed or rolled back. This prevents other statements from modifying the table for the entire duration of this transaction. This provides commit ordering and guarantees serializability across multiple transactions. - If in autocommit mode, or outside a transactional context, automatically release metadata locks of the current statement. */ thd->mdl_context.release_transactional_locks(); } else if (! thd->in_sub_stmt) { thd->mdl_context.release_statement_locks(); } if (thd->variables.session_track_transaction_info > TX_TRACK_NONE) { ((Transaction_state_tracker *) thd->session_tracker.get_tracker(TRANSACTION_INFO_TRACKER)) ->add_trx_state_from_thd(thd); } #if defined(VALGRIND_DO_QUICK_LEAK_CHECK) // Get incremental leak reports, for easier leak hunting. // ./mtr --mem --mysqld='-T 4096' --valgrind-mysqld main.1st // Note that with multiple connections, the report below may be misleading. if (test_flags & TEST_DO_QUICK_LEAK_CHECK) { static unsigned long total_leaked_bytes= 0; unsigned long leaked= 0; unsigned long dubious MY_ATTRIBUTE((unused)); unsigned long reachable MY_ATTRIBUTE((unused)); unsigned long suppressed MY_ATTRIBUTE((unused)); /* We could possibly use VALGRIND_DO_CHANGED_LEAK_CHECK here, but that is a fairly new addition to the Valgrind api. Note: we dont want to check 'reachable' until we have done shutdown, and that is handled by the final report anyways. We print some extra information, to tell mtr to ignore this report. */ sql_print_information("VALGRIND_DO_QUICK_LEAK_CHECK"); VALGRIND_DO_QUICK_LEAK_CHECK; VALGRIND_COUNT_LEAKS(leaked, dubious, reachable, suppressed); if (leaked > total_leaked_bytes) { sql_print_error("VALGRIND_COUNT_LEAKS reports %lu leaked bytes " "for query '%.*s'", leaked - total_leaked_bytes, static_cast<int>(thd->query().length), thd->query().str); } total_leaked_bytes= leaked; } #endif if (!res && !thd->is_error()) { // if statement succeeded binlog_gtid_end_transaction(thd); // finalize GTID life-cycle DEBUG_SYNC(thd, "persist_new_state_after_statement_succeeded"); } else if (!gtid_consistency_violation_state && // if the consistency state thd->has_gtid_consistency_violation) { // was set by the failing // statement gtid_state->end_gtid_violating_transaction(thd); // just roll it back DEBUG_SYNC(thd, "restore_previous_state_after_statement_failed"); } thd->skip_gtid_rollback = false; DBUG_RETURN(res || thd->is_error()); } bool execute_sqlcom_select(THD *thd, TABLE_LIST *all_tables) { LEX *lex= thd->lex; bool statement_timer_armed= false; bool res; /* assign global limit variable if limit is not given */ { SELECT_LEX *param= lex->unit->global_parameters(); if (!param->explicit_limit) param->select_limit= new Item_int((ulonglong) thd->variables.select_limit); } //check if timer is applicable to statement, if applicable then set timer. if (is_timer_applicable_to_statement(thd)) statement_timer_armed= set_statement_timer(thd); if (!(res= open_tables_for_query(thd, all_tables, 0))) { MYSQL_SELECT_START(const_cast<char*>(thd->query().str)); if (lex->is_explain()) { /* We always use Query_result_send for EXPLAIN, even if it's an EXPLAIN for SELECT ... INTO OUTFILE: a user application should be able to prepend EXPLAIN to any query and receive output for it, even if the query itself redirects the output. */ Query_result *const result= new Query_result_send; if (!result) return true; /* purecov: inspected */ res= handle_query(thd, lex, result, 0, 0); } else { Query_result *result= lex->result; if (!result && !(result= new Query_result_send())) return true; /* purecov: inspected */ Query_result *save_result= result; Query_result *analyse_result= NULL; if (lex->proc_analyse) { if ((result= analyse_result= new Query_result_analyse(result, lex->proc_analyse)) == NULL) return true; } res= handle_query(thd, lex, result, 0, 0); delete analyse_result; if (save_result != lex->result) delete save_result; } MYSQL_SELECT_DONE((int) res, (ulong) thd->current_found_rows); } if (statement_timer_armed && thd->timer) reset_statement_timer(thd); DEBUG_SYNC(thd, "after_table_open"); return res; } /**************************************************************************** Check stack size; Send error if there isn't enough stack to continue ****************************************************************************/ #if STACK_DIRECTION < 0 #define used_stack(A,B) (long) (A - B) #else #define used_stack(A,B) (long) (B - A) #endif #ifndef NDEBUG long max_stack_used; #endif /** @note Note: The 'buf' parameter is necessary, even if it is unused here. - fix_fields functions has a "dummy" buffer large enough for the corresponding exec. (Thus we only have to check in fix_fields.) - Passing to check_stack_overrun() prevents the compiler from removing it. */ bool check_stack_overrun(THD *thd, long margin, uchar *buf MY_ATTRIBUTE((unused))) { long stack_used; assert(thd == current_thd); if ((stack_used=used_stack(thd->thread_stack,(char*) &stack_used)) >= (long) (my_thread_stack_size - margin)) { /* Do not use stack for the message buffer to ensure correct behaviour in cases we have close to no stack left. */ char* ebuff= new (std::nothrow) char[MYSQL_ERRMSG_SIZE]; if (ebuff) { my_snprintf(ebuff, MYSQL_ERRMSG_SIZE, ER(ER_STACK_OVERRUN_NEED_MORE), stack_used, my_thread_stack_size, margin); my_message(ER_STACK_OVERRUN_NEED_MORE, ebuff, MYF(ME_FATALERROR)); delete [] ebuff; } return 1; } #ifndef NDEBUG max_stack_used= max(max_stack_used, stack_used); #endif return 0; } #define MY_YACC_INIT 1000 // Start with big alloc #define MY_YACC_MAX 32000 // Because of 'short' bool my_yyoverflow(short **yyss, YYSTYPE **yyvs, YYLTYPE **yyls, ulong *yystacksize) { Yacc_state *state= & current_thd->m_parser_state->m_yacc; ulong old_info=0; assert(state); if ((uint) *yystacksize >= MY_YACC_MAX) return 1; if (!state->yacc_yyvs) old_info= *yystacksize; *yystacksize= set_zone((*yystacksize)*2,MY_YACC_INIT,MY_YACC_MAX); if (!(state->yacc_yyvs= (uchar*) my_realloc(key_memory_bison_stack, state->yacc_yyvs, *yystacksize*sizeof(**yyvs), MYF(MY_ALLOW_ZERO_PTR | MY_FREE_ON_ERROR))) || !(state->yacc_yyss= (uchar*) my_realloc(key_memory_bison_stack, state->yacc_yyss, *yystacksize*sizeof(**yyss), MYF(MY_ALLOW_ZERO_PTR | MY_FREE_ON_ERROR))) || !(state->yacc_yyls= (uchar*) my_realloc(key_memory_bison_stack, state->yacc_yyls, *yystacksize*sizeof(**yyls), MYF(MY_ALLOW_ZERO_PTR | MY_FREE_ON_ERROR)))) return 1; if (old_info) { /* Only copy the old stack on the first call to my_yyoverflow(), when replacing a static stack (YYINITDEPTH) by a dynamic stack. For subsequent calls, my_realloc already did preserve the old stack. */ memcpy(state->yacc_yyss, *yyss, old_info*sizeof(**yyss)); memcpy(state->yacc_yyvs, *yyvs, old_info*sizeof(**yyvs)); memcpy(state->yacc_yyls, *yyls, old_info*sizeof(**yyls)); } *yyss= (short*) state->yacc_yyss; *yyvs= (YYSTYPE*) state->yacc_yyvs; *yyls= (YYLTYPE*) state->yacc_yyls; return 0; } /** Reset the part of THD responsible for the state of command processing. This needs to be called before execution of every statement (prepared or conventional). It is not called by substatements of routines. @todo Remove mysql_reset_thd_for_next_command and only use the member function. @todo Call it after we use THD for queries, not before. */ void mysql_reset_thd_for_next_command(THD *thd) { thd->reset_for_next_command(); } void THD::reset_for_next_command() { // TODO: Why on earth is this here?! We should probably fix this // function and move it to the proper file. /Matz THD *thd= this; DBUG_ENTER("mysql_reset_thd_for_next_command"); assert(!thd->sp_runtime_ctx); /* not for substatements of routines */ assert(! thd->in_sub_stmt); thd->free_list= 0; /* Those two lines below are theoretically unneeded as THD::cleanup_after_query() should take care of this already. */ thd->auto_inc_intervals_in_cur_stmt_for_binlog.empty(); thd->stmt_depends_on_first_successful_insert_id_in_prev_stmt= 0; thd->query_start_usec_used= 0; thd->is_fatal_error= thd->time_zone_used= 0; /* Clear the status flag that are expected to be cleared at the beginning of each SQL statement. */ thd->server_status&= ~SERVER_STATUS_CLEAR_SET; /* If in autocommit mode and not in a transaction, reset flag that identifies if a transaction has done some operations that cannot be safely rolled back. If the flag is set an warning message is printed out in ha_rollback_trans() saying that some tables couldn't be rolled back. */ if (!thd->in_multi_stmt_transaction_mode()) { thd->get_transaction()->reset_unsafe_rollback_flags( Transaction_ctx::SESSION); } assert(thd->security_context()== &thd->m_main_security_ctx); thd->thread_specific_used= FALSE; if (opt_bin_log) { thd->user_var_events.clear(); thd->user_var_events_alloc= thd->mem_root; } thd->clear_error(); thd->get_stmt_da()->reset_diagnostics_area(); thd->get_stmt_da()->reset_statement_cond_count(); thd->rand_used= 0; thd->m_sent_row_count= thd->m_examined_row_count= 0; thd->reset_current_stmt_binlog_format_row(); thd->binlog_unsafe_warning_flags= 0; thd->binlog_need_explicit_defaults_ts= false; thd->commit_error= THD::CE_NONE; thd->durability_property= HA_REGULAR_DURABILITY; thd->set_trans_pos(NULL, 0); thd->derived_tables_processing= false; // Need explicit setting, else demand all privileges to a table. thd->want_privilege= ~NO_ACCESS; thd->gtid_executed_warning_issued= false; thd->reset_skip_readonly_check(); DBUG_PRINT("debug", ("is_current_stmt_binlog_format_row(): %d", thd->is_current_stmt_binlog_format_row())); DBUG_VOID_RETURN; } /** Create a select to return the same output as 'SELECT @@var_name'. Used for SHOW COUNT(*) [ WARNINGS | ERROR]. This will crash with a core dump if the variable doesn't exists. @param pc Current parse context @param var_name Variable name */ void create_select_for_variable(Parse_context *pc, const char *var_name) { THD *thd= pc->thd; LEX *lex; LEX_STRING tmp, null_lex_string; Item *var; char buff[MAX_SYS_VAR_LENGTH*2+4+8], *end; DBUG_ENTER("create_select_for_variable"); lex= thd->lex; lex->sql_command= SQLCOM_SELECT; tmp.str= (char*) var_name; tmp.length=strlen(var_name); memset(&null_lex_string, 0, sizeof(null_lex_string)); /* We set the name of Item to @@session.var_name because that then is used as the column name in the output. */ if ((var= get_system_var(pc, OPT_SESSION, tmp, null_lex_string, false))) { end= strxmov(buff, "@@session.", var_name, NullS); var->item_name.copy(buff, end - buff); add_item_to_list(thd, var); } DBUG_VOID_RETURN; } void mysql_init_multi_delete(LEX *lex) { lex->select_lex->select_limit= 0; lex->unit->select_limit_cnt= HA_POS_ERROR; lex->select_lex->table_list.save_and_clear(&lex->auxiliary_table_list); lex->query_tables= NULL; lex->query_tables_last= &lex->query_tables; } /* When you modify mysql_parse(), you may need to mofify mysql_test_parse_for_slave() in this same file. */ /** Parse a query. @param thd Current thread @param rawbuf Begining of the query text @param length Length of the query text @param[out] found_semicolon For multi queries, position of the character of the next query in the query text. */ void mysql_parse(THD *thd, Parser_state *parser_state) { int error MY_ATTRIBUTE((unused)); DBUG_ENTER("mysql_parse"); DBUG_PRINT("mysql_parse", ("query: '%s'", thd->query().str)); DBUG_EXECUTE_IF("parser_debug", turn_parser_debug_on();); /* Warning. The purpose of query_cache_send_result_to_client() is to lookup the query in the query cache first, to avoid parsing and executing it. So, the natural implementation would be to: - first, call query_cache_send_result_to_client, - second, if caching failed, initialise the lexical and syntactic parser. The problem is that the query cache depends on a clean initialization of (among others) lex->safe_to_cache_query and thd->server_status, which are reset respectively in - lex_start() - mysql_reset_thd_for_next_command() So, initializing the lexical analyser *before* using the query cache is required for the cache to work properly. FIXME: cleanup the dependencies in the code to simplify this. */ mysql_reset_thd_for_next_command(thd); // It is possible that rewritten query may not be empty (in case of // multiqueries). So reset it. thd->reset_rewritten_query(); lex_start(thd); thd->m_parser_state= parser_state; invoke_pre_parse_rewrite_plugins(thd); thd->m_parser_state= NULL; enable_digest_if_any_plugin_needs_it(thd, parser_state); if (query_cache.send_result_to_client(thd, thd->query()) <= 0) { LEX *lex= thd->lex; const char *found_semicolon; size_t qlen = 0; bool err= thd->get_stmt_da()->is_error(); if (!err) { err= parse_sql(thd, parser_state, NULL); if (!err) err= invoke_post_parse_rewrite_plugins(thd, false); found_semicolon= parser_state->m_lip.found_semicolon; qlen = found_semicolon ? (found_semicolon - thd->query().str) : thd->query().length; /* We set thd->query_length correctly to not log several queries, when we execute only first. We set it to not see the ';' otherwise it would get into binlog and Query_log_event::print() would give ';;' output. */ if(!thd->is_error() && found_semicolon && (ulong)(qlen)) thd->set_query(thd->query().str, qlen - 1); } DEBUG_SYNC_C("sql_parse_before_rewrite"); if (!err) { /* Rewrite the query for logging and for the Performance Schema statement tables. (Raw logging happened earlier.) Query-cache only handles SELECT, which we don't rewrite, so it's no concern of ours. Sub-routines of mysql_rewrite_query() should try to only rewrite when necessary (e.g. not do password obfuscation when query contains no password). If rewriting does not happen here, thd->m_rewritten_query is still empty from being reset in alloc_query(). */ if (thd->rewritten_query().length() == 0) mysql_rewrite_query(thd); if (thd->rewritten_query().length()) { lex->safe_to_cache_query= false; // see comments below thd->set_query_for_display(thd->rewritten_query().ptr(), thd->rewritten_query().length()); } else if (thd->slave_thread) { /* In the slave, we add the information to pfs.events_statements_history, but not to pfs.threads, as that is what the test suite expects. */ MYSQL_SET_STATEMENT_TEXT(thd->m_statement_psi, thd->query().str, thd->query().length); } else { thd->set_query_for_display(thd->query().str, thd->query().length); } if (!(opt_general_log_raw || thd->slave_thread)) { if (thd->rewritten_query().length()) query_logger.general_log_write(thd, COM_QUERY, thd->rewritten_query().ptr(), thd->rewritten_query().length()); else { query_logger.general_log_write(thd, COM_QUERY, thd->query().str, qlen); } } } DEBUG_SYNC_C("sql_parse_after_rewrite"); if (!err) { thd->m_statement_psi= MYSQL_REFINE_STATEMENT(thd->m_statement_psi, sql_statement_info[thd->lex->sql_command].m_key); #ifndef NO_EMBEDDED_ACCESS_CHECKS if (mqh_used && thd->get_user_connect() && check_mqh(thd, lex->sql_command)) { if (thd->is_classic_protocol()) thd->get_protocol_classic()->get_net()->error = 0; } else #endif { if (!thd->is_error()) { /* Actually execute the query */ if (found_semicolon) { lex->safe_to_cache_query= 0; thd->server_status|= SERVER_MORE_RESULTS_EXISTS; } lex->set_trg_event_type_for_tables(); MYSQL_QUERY_EXEC_START( const_cast<char*>(thd->query().str), thd->thread_id(), (char *) (thd->db().str ? thd->db().str : ""), (char *) thd->security_context()->priv_user().str, (char *) thd->security_context()->host_or_ip().str, 0); if (unlikely(thd->security_context()->password_expired() && !lex->is_set_password_sql && lex->sql_command != SQLCOM_SET_OPTION && lex->sql_command != SQLCOM_ALTER_USER)) { my_error(ER_MUST_CHANGE_PASSWORD, MYF(0)); error= 1; } else error= mysql_execute_command(thd, true); MYSQL_QUERY_EXEC_DONE(error); } } } else { /* Log the failed raw query in the Performance Schema. This statement did not parse, so there is no way to tell if it may contain a password of not. The tradeoff is: a) If we do log the query, a user typing by accident a broken query containing a password will have the password exposed. This is very unlikely, and this behavior can be documented. Remediation is to use a new password when retyping the corrected query. b) If we do not log the query, finding broken queries in the client application will be much more difficult. This is much more likely. Considering that broken queries can typically be generated by attempts at SQL injection, finding the source of the SQL injection is critical, so the design choice is to log the query text of broken queries (a). */ thd->set_query_for_display(thd->query().str, thd->query().length); /* Instrument this broken statement as "statement/sql/error" */ thd->m_statement_psi= MYSQL_REFINE_STATEMENT(thd->m_statement_psi, sql_statement_info[SQLCOM_END].m_key); assert(thd->is_error()); DBUG_PRINT("info",("Command aborted. Fatal_error: %d", thd->is_fatal_error)); query_cache.abort(&thd->query_cache_tls); } THD_STAGE_INFO(thd, stage_freeing_items); sp_cache_enforce_limit(thd->sp_proc_cache, stored_program_cache_size); sp_cache_enforce_limit(thd->sp_func_cache, stored_program_cache_size); thd->end_statement(); thd->cleanup_after_query(); assert(thd->change_list.is_empty()); } else { /* Query cache hit. We need to write the general log here if we haven't already logged the statement earlier due to --log-raw. Right now, we only cache SELECT results; if the cache ever becomes more generic, we should also cache the rewritten query-string together with the original query-string (which we'd still use for the matching) when we first execute the query, and then use the obfuscated query-string for logging here when the query is given again. */ if (!opt_general_log_raw) query_logger.general_log_write(thd, COM_QUERY, thd->query().str, thd->query().length); parser_state->m_lip.found_semicolon= NULL; } DEBUG_SYNC(thd, "query_rewritten"); DBUG_VOID_RETURN; } #ifdef HAVE_REPLICATION /** Usable by the replication SQL thread only: just parse a query to know if it can be ignored because of replicate-*-table rules. @retval 0 cannot be ignored @retval 1 can be ignored */ bool mysql_test_parse_for_slave(THD *thd) { LEX *lex= thd->lex; bool ignorable= false; sql_digest_state *parent_digest= thd->m_digest; PSI_statement_locker *parent_locker= thd->m_statement_psi; DBUG_ENTER("mysql_test_parse_for_slave"); assert(thd->slave_thread); Parser_state parser_state; if (parser_state.init(thd, thd->query().str, thd->query().length) == 0) { lex_start(thd); mysql_reset_thd_for_next_command(thd); thd->m_digest= NULL; thd->m_statement_psi= NULL; if (parse_sql(thd, & parser_state, NULL) == 0) { if (all_tables_not_ok(thd, lex->select_lex->table_list.first)) ignorable= true; else if (!check_database_filters(thd, thd->db().str, lex->sql_command)) ignorable= true; } thd->m_digest= parent_digest; thd->m_statement_psi= parent_locker; thd->end_statement(); } thd->cleanup_after_query(); DBUG_RETURN(ignorable); } #endif /** Store field definition for create. @return Return 0 if ok */ bool add_field_to_list(THD *thd, LEX_STRING *field_name, enum_field_types type, char *length, char *decimals, uint type_modifier, Item *default_value, Item *on_update_value, LEX_STRING *comment, char *change, List<String> *interval_list, const CHARSET_INFO *cs, uint uint_geom_type, Generated_column *gcol_info) { Create_field *new_field; LEX *lex= thd->lex; uint8 datetime_precision= decimals ? atoi(decimals) : 0; DBUG_ENTER("add_field_to_list"); LEX_CSTRING field_name_cstr= {field_name->str, field_name->length}; if (check_string_char_length(field_name_cstr, "", NAME_CHAR_LEN, system_charset_info, 1)) { my_error(ER_TOO_LONG_IDENT, MYF(0), field_name->str); /* purecov: inspected */ DBUG_RETURN(1); /* purecov: inspected */ } if (type_modifier & PRI_KEY_FLAG) { Key *key; lex->col_list.push_back(new Key_part_spec(*field_name, 0)); key= new Key(KEYTYPE_PRIMARY, null_lex_str, &default_key_create_info, 0, lex->col_list); lex->alter_info.key_list.push_back(key); lex->col_list.empty(); } if (type_modifier & (UNIQUE_FLAG | UNIQUE_KEY_FLAG)) { Key *key; lex->col_list.push_back(new Key_part_spec(*field_name, 0)); key= new Key(KEYTYPE_UNIQUE, null_lex_str, &default_key_create_info, 0, lex->col_list); lex->alter_info.key_list.push_back(key); lex->col_list.empty(); } if (default_value) { /* Default value should be literal => basic constants => no need fix_fields() We allow only CURRENT_TIMESTAMP as function default for the TIMESTAMP or DATETIME types. */ if (default_value->type() == Item::FUNC_ITEM && (static_cast<Item_func*>(default_value)->functype() != Item_func::NOW_FUNC || (!real_type_with_now_as_default(type)) || default_value->decimals != datetime_precision)) { my_error(ER_INVALID_DEFAULT, MYF(0), field_name->str); DBUG_RETURN(1); } else if (default_value->type() == Item::NULL_ITEM) { default_value= 0; if ((type_modifier & (NOT_NULL_FLAG | AUTO_INCREMENT_FLAG)) == NOT_NULL_FLAG) { my_error(ER_INVALID_DEFAULT, MYF(0), field_name->str); DBUG_RETURN(1); } } else if (type_modifier & AUTO_INCREMENT_FLAG) { my_error(ER_INVALID_DEFAULT, MYF(0), field_name->str); DBUG_RETURN(1); } } if (on_update_value && (!real_type_with_now_on_update(type) || on_update_value->decimals != datetime_precision)) { my_error(ER_INVALID_ON_UPDATE, MYF(0), field_name->str); DBUG_RETURN(1); } if (!(new_field= new Create_field()) || new_field->init(thd, field_name->str, type, length, decimals, type_modifier, default_value, on_update_value, comment, change, interval_list, cs, uint_geom_type, gcol_info)) DBUG_RETURN(1); lex->alter_info.create_list.push_back(new_field); lex->last_field=new_field; DBUG_RETURN(0); } /** Store position for column in ALTER TABLE .. ADD column. */ void store_position_for_column(const char *name) { current_thd->lex->last_field->after=(char*) (name); } /** save order by and tables in own lists. */ void add_to_list(SQL_I_List<ORDER> &list, ORDER *order) { DBUG_ENTER("add_to_list"); order->item= &order->item_ptr; order->used_alias= false; order->used=0; list.link_in_list(order, &order->next); DBUG_VOID_RETURN; } /** Add a table to list of used tables. @param table Table to add @param alias alias for table (or null if no alias) @param table_options A set of the following bits: - TL_OPTION_UPDATING : Table will be updated - TL_OPTION_FORCE_INDEX : Force usage of index - TL_OPTION_ALIAS : an alias in multi table DELETE @param lock_type How table should be locked @param mdl_type Type of metadata lock to acquire on the table. @param use_index List of indexed used in USE INDEX @param ignore_index List of indexed used in IGNORE INDEX @retval 0 Error @retval \# Pointer to TABLE_LIST element added to the total table list */ TABLE_LIST *st_select_lex::add_table_to_list(THD *thd, Table_ident *table, LEX_STRING *alias, ulong table_options, thr_lock_type lock_type, enum_mdl_type mdl_type, List<Index_hint> *index_hints_arg, List<String> *partition_names, LEX_STRING *option) { TABLE_LIST *ptr; TABLE_LIST *previous_table_ref= NULL; /* The table preceding the current one. */ const char *alias_str; LEX *lex= thd->lex; DBUG_ENTER("add_table_to_list"); if (!table) DBUG_RETURN(0); // End of memory alias_str= alias ? alias->str : table->table.str; if (!MY_TEST(table_options & TL_OPTION_ALIAS)) { enum_ident_name_check ident_check_status= check_table_name(table->table.str, table->table.length, FALSE); if (ident_check_status == IDENT_NAME_WRONG) { my_error(ER_WRONG_TABLE_NAME, MYF(0), table->table.str); DBUG_RETURN(0); } else if (ident_check_status == IDENT_NAME_TOO_LONG) { my_error(ER_TOO_LONG_IDENT, MYF(0), table->table.str); DBUG_RETURN(0); } } LEX_STRING db= to_lex_string(table->db); if (table->is_derived_table() == FALSE && table->db.str && (check_and_convert_db_name(&db, FALSE) != IDENT_NAME_OK)) DBUG_RETURN(0); if (!alias) /* Alias is case sensitive */ { if (table->sel) { my_message(ER_DERIVED_MUST_HAVE_ALIAS, ER(ER_DERIVED_MUST_HAVE_ALIAS), MYF(0)); DBUG_RETURN(0); } if (!(alias_str= (char*) thd->memdup(alias_str,table->table.length+1))) DBUG_RETURN(0); } if (!(ptr = (TABLE_LIST *) thd->mem_calloc(sizeof(TABLE_LIST)))) DBUG_RETURN(0); /* purecov: inspected */ if (table->db.str) { ptr->is_fqtn= TRUE; ptr->db= const_cast<char*>(table->db.str); ptr->db_length= table->db.length; } else if (lex->copy_db_to((char**)&ptr->db, &ptr->db_length)) DBUG_RETURN(0); else ptr->is_fqtn= FALSE; ptr->alias= const_cast<char*>(alias_str); ptr->is_alias= alias ? TRUE : FALSE; if (lower_case_table_names && table->table.length) table->table.length= my_casedn_str(files_charset_info, const_cast<char*>(table->table.str)); ptr->table_name= const_cast<char*>(table->table.str); ptr->table_name_length= table->table.length; ptr->set_tableno(0); ptr->lock_type= lock_type; ptr->updating= MY_TEST(table_options & TL_OPTION_UPDATING); /* TODO: remove TL_OPTION_FORCE_INDEX as it looks like it's not used */ ptr->force_index= MY_TEST(table_options & TL_OPTION_FORCE_INDEX); ptr->ignore_leaves= MY_TEST(table_options & TL_OPTION_IGNORE_LEAVES); ptr->set_derived_unit(table->sel); if (!ptr->is_derived() && is_infoschema_db(ptr->db, ptr->db_length)) { ST_SCHEMA_TABLE *schema_table; if (ptr->updating && /* Special cases which are processed by commands itself */ lex->sql_command != SQLCOM_CHECK && lex->sql_command != SQLCOM_CHECKSUM) { my_error(ER_DBACCESS_DENIED_ERROR, MYF(0), thd->security_context()->priv_user().str, thd->security_context()->priv_host().str, INFORMATION_SCHEMA_NAME.str); DBUG_RETURN(0); } schema_table= find_schema_table(thd, ptr->table_name); if (!schema_table || (schema_table->hidden && ((sql_command_flags[lex->sql_command] & CF_STATUS_COMMAND) == 0 || /* this check is used for show columns|keys from I_S hidden table */ lex->sql_command == SQLCOM_SHOW_FIELDS || lex->sql_command == SQLCOM_SHOW_KEYS))) { my_error(ER_UNKNOWN_TABLE, MYF(0), ptr->table_name, INFORMATION_SCHEMA_NAME.str); DBUG_RETURN(0); } ptr->schema_table_name= const_cast<char*>(ptr->table_name); ptr->schema_table= schema_table; } ptr->select_lex= this; ptr->cacheable_table= 1; ptr->index_hints= index_hints_arg; ptr->option= option ? option->str : 0; /* check that used name is unique */ if (lock_type != TL_IGNORE) { TABLE_LIST *first_table= table_list.first; if (lex->sql_command == SQLCOM_CREATE_VIEW) first_table= first_table ? first_table->next_local : NULL; for (TABLE_LIST *tables= first_table ; tables ; tables=tables->next_local) { if (!my_strcasecmp(table_alias_charset, alias_str, tables->alias) && !strcmp(ptr->db, tables->db)) { my_error(ER_NONUNIQ_TABLE, MYF(0), alias_str); /* purecov: tested */ DBUG_RETURN(0); /* purecov: tested */ } } } /* Store the table reference preceding the current one. */ if (table_list.elements > 0) { /* table_list.next points to the last inserted TABLE_LIST->next_local' element We don't use the offsetof() macro here to avoid warnings from gcc */ previous_table_ref= (TABLE_LIST*) ((char*) table_list.next - ((char*) &(ptr->next_local) - (char*) ptr)); /* Set next_name_resolution_table of the previous table reference to point to the current table reference. In effect the list TABLE_LIST::next_name_resolution_table coincides with TABLE_LIST::next_local. Later this may be changed in store_top_level_join_columns() for NATURAL/USING joins. */ previous_table_ref->next_name_resolution_table= ptr; } /* Link the current table reference in a local list (list for current select). Notice that as a side effect here we set the next_local field of the previous table reference to 'ptr'. Here we also add one element to the list 'table_list'. */ table_list.link_in_list(ptr, &ptr->next_local); ptr->next_name_resolution_table= NULL; ptr->partition_names= partition_names; /* Link table in global list (all used tables) */ lex->add_to_query_tables(ptr); // Pure table aliases do not need to be locked: if (!MY_TEST(table_options & TL_OPTION_ALIAS)) { MDL_REQUEST_INIT(& ptr->mdl_request, MDL_key::TABLE, ptr->db, ptr->table_name, mdl_type, MDL_TRANSACTION); } if (table->is_derived_table()) { ptr->derived_key_list.empty(); derived_table_count++; } DBUG_RETURN(ptr); } /** Initialize a new table list for a nested join. The function initializes a structure of the TABLE_LIST type for a nested join. It sets up its nested join list as empty. The created structure is added to the front of the current join list in the st_select_lex object. Then the function changes the current nest level for joins to refer to the newly created empty list after having saved the info on the old level in the initialized structure. @param thd current thread @retval 0 if success @retval 1 otherwise */ bool st_select_lex::init_nested_join(THD *thd) { DBUG_ENTER("init_nested_join"); TABLE_LIST *const ptr= TABLE_LIST::new_nested_join(thd->mem_root, "(nested_join)", embedding, join_list, this); if (ptr == NULL) DBUG_RETURN(true); join_list->push_front(ptr); embedding= ptr; join_list= &ptr->nested_join->join_list; DBUG_RETURN(false); } /** End a nested join table list. The function returns to the previous join nest level. If the current level contains only one member, the function moves it one level up, eliminating the nest. @param thd current thread @return - Pointer to TABLE_LIST element added to the total table list, if success - 0, otherwise */ TABLE_LIST *st_select_lex::end_nested_join(THD *thd) { TABLE_LIST *ptr; NESTED_JOIN *nested_join; DBUG_ENTER("end_nested_join"); assert(embedding); ptr= embedding; join_list= ptr->join_list; embedding= ptr->embedding; nested_join= ptr->nested_join; if (nested_join->join_list.elements == 1) { TABLE_LIST *embedded= nested_join->join_list.head(); join_list->pop(); embedded->join_list= join_list; embedded->embedding= embedding; join_list->push_front(embedded); ptr= embedded; } else if (nested_join->join_list.elements == 0) { join_list->pop(); ptr= 0; // return value } DBUG_RETURN(ptr); } /** Nest last join operation. The function nest last join operation as if it was enclosed in braces. @param thd current thread @retval 0 Error @retval \# Pointer to TABLE_LIST element created for the new nested join */ TABLE_LIST *st_select_lex::nest_last_join(THD *thd) { DBUG_ENTER("nest_last_join"); TABLE_LIST *const ptr= TABLE_LIST::new_nested_join(thd->mem_root, "(nest_last_join)", embedding, join_list, this); if (ptr == NULL) DBUG_RETURN(NULL); List<TABLE_LIST> *const embedded_list= &ptr->nested_join->join_list; for (uint i=0; i < 2; i++) { TABLE_LIST *table= join_list->pop(); table->join_list= embedded_list; table->embedding= ptr; embedded_list->push_back(table); if (table->natural_join) { ptr->is_natural_join= TRUE; /* If this is a JOIN ... USING, move the list of joined fields to the table reference that describes the join. */ if (prev_join_using) ptr->join_using_fields= prev_join_using; } } join_list->push_front(ptr); DBUG_RETURN(ptr); } /** Add a table to the current join list. The function puts a table in front of the current join list of st_select_lex object. Thus, joined tables are put into this list in the reverse order (the most outer join operation follows first). @param table the table to add @return None */ void st_select_lex::add_joined_table(TABLE_LIST *table) { DBUG_ENTER("add_joined_table"); join_list->push_front(table); table->join_list= join_list; table->embedding= embedding; DBUG_VOID_RETURN; } /** Convert a right join into equivalent left join. The function takes the current join list t[0],t[1] ... and effectively converts it into the list t[1],t[0] ... Although the outer_join flag for the new nested table contains JOIN_TYPE_RIGHT, it will be handled as the inner table of a left join operation. EXAMPLES @verbatim SELECT * FROM t1 RIGHT JOIN t2 ON on_expr => SELECT * FROM t2 LEFT JOIN t1 ON on_expr SELECT * FROM t1,t2 RIGHT JOIN t3 ON on_expr => SELECT * FROM t1,t3 LEFT JOIN t2 ON on_expr SELECT * FROM t1,t2 RIGHT JOIN (t3,t4) ON on_expr => SELECT * FROM t1,(t3,t4) LEFT JOIN t2 ON on_expr SELECT * FROM t1 LEFT JOIN t2 ON on_expr1 RIGHT JOIN t3 ON on_expr2 => SELECT * FROM t3 LEFT JOIN (t1 LEFT JOIN t2 ON on_expr2) ON on_expr1 @endverbatim @param thd current thread @return - Pointer to the table representing the inner table, if success - 0, otherwise */ TABLE_LIST *st_select_lex::convert_right_join() { TABLE_LIST *tab2= join_list->pop(); TABLE_LIST *tab1= join_list->pop(); DBUG_ENTER("convert_right_join"); join_list->push_front(tab2); join_list->push_front(tab1); tab1->outer_join|= JOIN_TYPE_RIGHT; DBUG_RETURN(tab1); } /** Set lock for all tables in current select level. @param lock_type Lock to set for tables @note If lock is a write lock, then tables->updating is set 1 This is to get tables_ok to know that the table is updated by the query Set type of metadata lock to request according to lock_type. */ void st_select_lex::set_lock_for_tables(thr_lock_type lock_type) { bool for_update= lock_type >= TL_READ_NO_INSERT; enum_mdl_type mdl_type= mdl_type_for_dml(lock_type); DBUG_ENTER("set_lock_for_tables"); DBUG_PRINT("enter", ("lock_type: %d for_update: %d", lock_type, for_update)); for (TABLE_LIST *tables= table_list.first; tables; tables= tables->next_local) { tables->lock_type= lock_type; tables->updating= for_update; tables->mdl_request.set_type(mdl_type); } DBUG_VOID_RETURN; } /** Create a fake SELECT_LEX for a unit. The method create a fake SELECT_LEX object for a unit. This object is created for any union construct containing a union operation and also for any single select union construct of the form @verbatim (SELECT ... ORDER BY order_list [LIMIT n]) ORDER BY ... @endvarbatim or of the form @varbatim (SELECT ... ORDER BY LIMIT n) ORDER BY ... @endvarbatim @param thd_arg thread handle @note The object is used to retrieve rows from the temporary table where the result on the union is obtained. @retval 1 on failure to create the object @retval 0 on success */ bool st_select_lex_unit::add_fake_select_lex(THD *thd_arg) { SELECT_LEX *first_sl= first_select(); DBUG_ENTER("add_fake_select_lex"); assert(!fake_select_lex); assert(thd_arg == thd); if (!(fake_select_lex= thd_arg->lex->new_empty_query_block())) DBUG_RETURN(true); /* purecov: inspected */ fake_select_lex->include_standalone(this, &fake_select_lex); fake_select_lex->select_number= INT_MAX; fake_select_lex->linkage= GLOBAL_OPTIONS_TYPE; fake_select_lex->select_limit= 0; fake_select_lex->set_context(first_sl->context.outer_context); /* allow item list resolving in fake select for ORDER BY */ fake_select_lex->context.resolve_in_select_list= TRUE; if (!is_union()) { /* This works only for (SELECT ... ORDER BY list [LIMIT n]) ORDER BY order_list [LIMIT m], (SELECT ... LIMIT n) ORDER BY order_list [LIMIT m] just before the parser starts processing order_list */ fake_select_lex->no_table_names_allowed= 1; } thd->lex->pop_context(); DBUG_RETURN(false); } /** Push a new name resolution context for a JOIN ... ON clause to the context stack of a query block. Create a new name resolution context for a JOIN ... ON clause, set the first and last leaves of the list of table references to be used for name resolution, and push the newly created context to the stack of contexts of the query. @param pc current parse context @param left_op left operand of the JOIN @param right_op rigth operand of the JOIN @todo Research if we should set the "outer_context" member of the new ON context. @retval FALSE if all is OK @retval TRUE if a memory allocation error occured */ bool push_new_name_resolution_context(Parse_context *pc, TABLE_LIST *left_op, TABLE_LIST *right_op) { THD *thd= pc->thd; Name_resolution_context *on_context; if (!(on_context= new (thd->mem_root) Name_resolution_context)) return TRUE; on_context->init(); on_context->first_name_resolution_table= left_op->first_leaf_for_name_resolution(); on_context->last_name_resolution_table= right_op->last_leaf_for_name_resolution(); on_context->select_lex= pc->select; on_context->next_context= pc->select->first_context; pc->select->first_context= on_context; return thd->lex->push_context(on_context); } /** Add an ON condition to the second operand of a JOIN ... ON. Add an ON condition to the right operand of a JOIN ... ON clause. @param b the second operand of a JOIN ... ON @param expr the condition to be added to the ON clause */ void add_join_on(TABLE_LIST *b, Item *expr) { if (expr) { b->set_join_cond_optim((Item*)1); // m_join_cond_optim is not ready if (!b->join_cond()) b->set_join_cond(expr); else { /* If called from the parser, this happens if you have both a right and left join. If called later, it happens if we add more than one condition to the ON clause. */ b->set_join_cond(new Item_cond_and(b->join_cond(), expr)); } b->join_cond()->top_level_item(); } } /** Mark that there is a NATURAL JOIN or JOIN ... USING between two tables. This function marks that table b should be joined with a either via a NATURAL JOIN or via JOIN ... USING. Both join types are special cases of each other, so we treat them together. The function setup_conds() creates a list of equal condition between all fields of the same name for NATURAL JOIN or the fields in 'using_fields' for JOIN ... USING. The list of equality conditions is stored either in b->join_cond(), or in JOIN::conds, depending on whether there was an outer join. EXAMPLE @verbatim SELECT * FROM t1 NATURAL LEFT JOIN t2 <=> SELECT * FROM t1 LEFT JOIN t2 ON (t1.i=t2.i and t1.j=t2.j ... ) SELECT * FROM t1 NATURAL JOIN t2 WHERE <some_cond> <=> SELECT * FROM t1, t2 WHERE (t1.i=t2.i and t1.j=t2.j and <some_cond>) SELECT * FROM t1 JOIN t2 USING(j) WHERE <some_cond> <=> SELECT * FROM t1, t2 WHERE (t1.j=t2.j and <some_cond>) @endverbatim @param a Left join argument @param b Right join argument @param using_fields Field names from USING clause */ void add_join_natural(TABLE_LIST *a, TABLE_LIST *b, List<String> *using_fields, SELECT_LEX *lex) { b->natural_join= a; lex->prev_join_using= using_fields; } /** kill on thread. @param thd Thread class @param id Thread id @param only_kill_query Should it kill the query or the connection @note This is written such that we have a short lock on LOCK_thd_list */ static uint kill_one_thread(THD *thd, my_thread_id id, bool only_kill_query) { THD *tmp= NULL; uint error=ER_NO_SUCH_THREAD; Find_thd_with_id find_thd_with_id(id); DBUG_ENTER("kill_one_thread"); DBUG_PRINT("enter", ("id=%u only_kill=%d", id, only_kill_query)); DEBUG_SYNC(thd, "kill_thd_begin"); tmp= Global_THD_manager::get_instance()->find_thd(&find_thd_with_id); if (tmp) { /* If we're SUPER, we can KILL anything, including system-threads. No further checks. KILLer: thd->m_security_ctx->user could in theory be NULL while we're still in "unauthenticated" state. This is a theoretical case (the code suggests this could happen, so we play it safe). KILLee: tmp->m_security_ctx->user will be NULL for system threads. We need to check so Jane Random User doesn't crash the server when trying to kill a) system threads or b) unauthenticated users' threads (Bug#43748). If user of both killer and killee are non-NULL, proceed with slayage if both are string-equal. */ if ((thd->security_context()->check_access(SUPER_ACL)) || thd->security_context()->user_matches(tmp->security_context())) { /* process the kill only if thread is not already undergoing any kill connection. */ if (tmp->killed != THD::KILL_CONNECTION) { tmp->awake(only_kill_query ? THD::KILL_QUERY : THD::KILL_CONNECTION); } error= 0; } else error=ER_KILL_DENIED_ERROR; mysql_mutex_unlock(&tmp->LOCK_thd_data); } DEBUG_SYNC(thd, "kill_thd_end"); DBUG_PRINT("exit", ("%d", error)); DBUG_RETURN(error); } /* kills a thread and sends response SYNOPSIS sql_kill() thd Thread class id Thread id only_kill_query Should it kill the query or the connection */ static void sql_kill(THD *thd, my_thread_id id, bool only_kill_query) { uint error; if (!(error= kill_one_thread(thd, id, only_kill_query))) { if (! thd->killed) my_ok(thd); } else my_error(error, MYF(0), id); } /** This class implements callback function used by killall_non_super_threads to kill all threads that do not have the SUPER privilege */ class Kill_non_super_conn : public Do_THD_Impl { private: /* THD of connected client. */ THD *m_client_thd; public: Kill_non_super_conn(THD *thd) : m_client_thd(thd) { assert(m_client_thd->security_context()->check_access(SUPER_ACL)); } virtual void operator()(THD *thd_to_kill) { mysql_mutex_lock(&thd_to_kill->LOCK_thd_data); /* Kill only if non super thread and non slave thread. If an account has not yet been assigned to the security context of the thread we cannot tell if the account is super user or not. In this case we cannot kill that thread. In offline mode, after the account is assigned to this thread and it turns out it is not super user thread, the authentication for this thread will fail and the thread will be terminated. */ if (thd_to_kill->security_context()->has_account_assigned() && !(thd_to_kill->security_context()->check_access(SUPER_ACL)) && thd_to_kill->killed != THD::KILL_CONNECTION && !thd_to_kill->slave_thread) thd_to_kill->awake(THD::KILL_CONNECTION); mysql_mutex_unlock(&thd_to_kill->LOCK_thd_data); } }; /* kills all the threads that do not have the SUPER privilege. SYNOPSIS killall_non_super_threads() thd Thread class */ void killall_non_super_threads(THD *thd) { Kill_non_super_conn kill_non_super_conn(thd); Global_THD_manager *thd_manager= Global_THD_manager::get_instance(); thd_manager->do_for_all_thd(&kill_non_super_conn); } /** If pointer is not a null pointer, append filename to it. */ bool append_file_to_dir(THD *thd, const char **filename_ptr, const char *table_name) { char buff[FN_REFLEN],*ptr, *end; if (!*filename_ptr) return 0; // nothing to do /* Check that the filename is not too long and it's a hard path */ if (strlen(*filename_ptr)+strlen(table_name) >= FN_REFLEN-1 || !test_if_hard_path(*filename_ptr)) { my_error(ER_WRONG_TABLE_NAME, MYF(0), *filename_ptr); return 1; } /* Fix is using unix filename format on dos */ my_stpcpy(buff,*filename_ptr); end=convert_dirname(buff, *filename_ptr, NullS); if (!(ptr= (char*) thd->alloc((size_t) (end-buff) + strlen(table_name)+1))) return 1; // End of memory *filename_ptr=ptr; strxmov(ptr,buff,table_name,NullS); return 0; } Comp_creator *comp_eq_creator(bool invert) { return invert?(Comp_creator *)&ne_creator:(Comp_creator *)&eq_creator; } Comp_creator *comp_equal_creator(bool invert) { assert(!invert); // Function never called with true. return &equal_creator; } Comp_creator *comp_ge_creator(bool invert) { return invert?(Comp_creator *)<_creator:(Comp_creator *)&ge_creator; } Comp_creator *comp_gt_creator(bool invert) { return invert?(Comp_creator *)&le_creator:(Comp_creator *)>_creator; } Comp_creator *comp_le_creator(bool invert) { return invert?(Comp_creator *)>_creator:(Comp_creator *)&le_creator; } Comp_creator *comp_lt_creator(bool invert) { return invert?(Comp_creator *)&ge_creator:(Comp_creator *)<_creator; } Comp_creator *comp_ne_creator(bool invert) { return invert?(Comp_creator *)&eq_creator:(Comp_creator *)&ne_creator; } /** Construct ALL/ANY/SOME subquery Item. @param left_expr pointer to left expression @param cmp compare function creator @param all true if we create ALL subquery @param select_lex pointer on parsed subquery structure @return constructed Item (or 0 if out of memory) */ Item * all_any_subquery_creator(Item *left_expr, chooser_compare_func_creator cmp, bool all, SELECT_LEX *select_lex) { if ((cmp == &comp_eq_creator) && !all) // = ANY <=> IN return new Item_in_subselect(left_expr, select_lex); if ((cmp == &comp_ne_creator) && all) // <> ALL <=> NOT IN return new Item_func_not(new Item_in_subselect(left_expr, select_lex)); Item_allany_subselect *it= new Item_allany_subselect(left_expr, cmp, select_lex, all); if (all) return it->upper_item= new Item_func_not_all(it); /* ALL */ return it->upper_item= new Item_func_nop_all(it); /* ANY/SOME */ } /** Set proper open mode and table type for element representing target table of CREATE TABLE statement, also adjust statement table list if necessary. */ void create_table_set_open_action_and_adjust_tables(LEX *lex) { TABLE_LIST *create_table= lex->query_tables; if (lex->create_info.options & HA_LEX_CREATE_TMP_TABLE) create_table->open_type= OT_TEMPORARY_ONLY; else create_table->open_type= OT_BASE_ONLY; if (!lex->select_lex->item_list.elements) { /* Avoid opening and locking target table for ordinary CREATE TABLE or CREATE TABLE LIKE for write (unlike in CREATE ... SELECT we won't do any insertions in it anyway). Not doing this causes problems when running CREATE TABLE IF NOT EXISTS for already existing log table. */ create_table->lock_type= TL_READ; } } /** negate given expression. @param pc current parse context @param expr expression for negation @return negated expression */ Item *negate_expression(Parse_context *pc, Item *expr) { Item *negated; if (expr->type() == Item::FUNC_ITEM && ((Item_func *) expr)->functype() == Item_func::NOT_FUNC) { /* it is NOT(NOT( ... )) */ Item *arg= ((Item_func *) expr)->arguments()[0]; enum_parsing_context place= pc->select->parsing_place; if (arg->is_bool_func() || place == CTX_WHERE || place == CTX_HAVING) return arg; /* if it is not boolean function then we have to emulate value of not(not(a)), it will be a != 0 */ return new Item_func_ne(arg, new Item_int_0()); } if ((negated= expr->neg_transformer(pc->thd)) != 0) return negated; return new Item_func_not(expr); } /** Set the specified definer to the default value, which is the current user in the thread. @param[in] thd thread handler @param[out] definer definer */ void get_default_definer(THD *thd, LEX_USER *definer) { const Security_context *sctx= thd->security_context(); definer->user.str= (char *) sctx->priv_user().str; definer->user.length= strlen(definer->user.str); definer->host.str= (char *) sctx->priv_host().str; definer->host.length= strlen(definer->host.str); definer->plugin= EMPTY_CSTR; definer->auth= NULL_CSTR; definer->uses_identified_with_clause= false; definer->uses_identified_by_clause= false; definer->uses_authentication_string_clause= false; definer->uses_identified_by_password_clause= false; definer->alter_status.update_password_expired_column= false; definer->alter_status.use_default_password_lifetime= true; definer->alter_status.expire_after_days= 0; definer->alter_status.update_account_locked_column= false; definer->alter_status.account_locked= false; } /** Create default definer for the specified THD. @param[in] thd thread handler @return - On success, return a valid pointer to the created and initialized LEX_USER, which contains definer information. - On error, return 0. */ LEX_USER *create_default_definer(THD *thd) { LEX_USER *definer; if (! (definer= (LEX_USER*) thd->alloc(sizeof(LEX_USER)))) return 0; thd->get_definer(definer); return definer; } /** Retuns information about user or current user. @param[in] thd thread handler @param[in] user user @return - On success, return a valid pointer to initialized LEX_USER, which contains user information. - On error, return 0. */ LEX_USER *get_current_user(THD *thd, LEX_USER *user) { if (!user->user.str) // current_user { LEX_USER *default_definer= create_default_definer(thd); if (default_definer) { /* Inherit parser semantics from the statement in which the user parameter was used. This is needed because a st_lex_user is both used as a component in an AST and as a specifier for a particular user in the ACL subsystem. */ default_definer->uses_authentication_string_clause= user->uses_authentication_string_clause; default_definer->uses_identified_by_clause= user->uses_identified_by_clause; default_definer->uses_identified_by_password_clause= user->uses_identified_by_password_clause; default_definer->uses_identified_with_clause= user->uses_identified_with_clause; default_definer->plugin.str= user->plugin.str; default_definer->plugin.length= user->plugin.length; default_definer->auth.str= user->auth.str; default_definer->auth.length= user->auth.length; default_definer->alter_status= user->alter_status; return default_definer; } } return user; } /** Check that byte length of a string does not exceed some limit. @param str string to be checked @param err_msg error message to be displayed if the string is too long @param max_length max length @retval FALSE the passed string is not longer than max_length @retval TRUE the passed string is longer than max_length NOTE The function is not used in existing code but can be useful later? */ static bool check_string_byte_length(const LEX_CSTRING &str, const char *err_msg, size_t max_byte_length) { if (str.length <= max_byte_length) return FALSE; my_error(ER_WRONG_STRING_LENGTH, MYF(0), str.str, err_msg, max_byte_length); return TRUE; } /* Check that char length of a string does not exceed some limit. SYNOPSIS check_string_char_length() str string to be checked err_msg error message to be displayed if the string is too long max_char_length max length in symbols cs string charset RETURN FALSE the passed string is not longer than max_char_length TRUE the passed string is longer than max_char_length */ bool check_string_char_length(const LEX_CSTRING &str, const char *err_msg, size_t max_char_length, const CHARSET_INFO *cs, bool no_error) { int well_formed_error; size_t res= cs->cset->well_formed_len(cs, str.str, str.str + str.length, max_char_length, &well_formed_error); if (!well_formed_error && str.length == res) return FALSE; if (!no_error) { ErrConvString err(str.str, str.length, cs); my_error(ER_WRONG_STRING_LENGTH, MYF(0), err.ptr(), err_msg, max_char_length); } return TRUE; } /* Check if path does not contain mysql data home directory SYNOPSIS test_if_data_home_dir() dir directory conv_home_dir converted data home directory home_dir_len converted data home directory length RETURN VALUES 0 ok 1 error */ C_MODE_START int test_if_data_home_dir(const char *dir) { char path[FN_REFLEN]; size_t dir_len; DBUG_ENTER("test_if_data_home_dir"); if (!dir) DBUG_RETURN(0); (void) fn_format(path, dir, "", "", (MY_RETURN_REAL_PATH|MY_RESOLVE_SYMLINKS)); dir_len= strlen(path); if (mysql_unpacked_real_data_home_len<= dir_len) { if (dir_len > mysql_unpacked_real_data_home_len && path[mysql_unpacked_real_data_home_len] != FN_LIBCHAR) DBUG_RETURN(0); if (lower_case_file_system) { if (!my_strnncoll(default_charset_info, (const uchar*) path, mysql_unpacked_real_data_home_len, (const uchar*) mysql_unpacked_real_data_home, mysql_unpacked_real_data_home_len)) DBUG_RETURN(1); } else if (!memcmp(path, mysql_unpacked_real_data_home, mysql_unpacked_real_data_home_len)) DBUG_RETURN(1); } DBUG_RETURN(0); } C_MODE_END /** Check that host name string is valid. @param[in] str string to be checked @return Operation status @retval FALSE host name is ok @retval TRUE host name string is longer than max_length or has invalid symbols */ bool check_host_name(const LEX_CSTRING &str) { const char *name= str.str; const char *end= str.str + str.length; if (check_string_byte_length(str, ER(ER_HOSTNAME), HOSTNAME_LENGTH)) return TRUE; while (name != end) { if (*name == '@') { my_printf_error(ER_UNKNOWN_ERROR, "Malformed hostname (illegal symbol: '%c')", MYF(0), *name); return TRUE; } name++; } return FALSE; } class Parser_oom_handler : public Internal_error_handler { public: Parser_oom_handler() : m_has_errors(false), m_is_mem_error(false) {} virtual bool handle_condition(THD *thd, uint sql_errno, const char* sqlstate, Sql_condition::enum_severity_level *level, const char* msg) { if (*level == Sql_condition::SL_ERROR) { m_has_errors= true; /* Out of memory error is reported only once. Return as handled */ if (m_is_mem_error && sql_errno == EE_CAPACITY_EXCEEDED) return true; if (sql_errno == EE_CAPACITY_EXCEEDED) { m_is_mem_error= true; my_error(ER_CAPACITY_EXCEEDED, MYF(0), static_cast<ulonglong>(thd->variables.parser_max_mem_size), "parser_max_mem_size", ER_THD(thd, ER_CAPACITY_EXCEEDED_IN_PARSER)); return true; } } return false; } private: bool m_has_errors; bool m_is_mem_error; }; extern int MYSQLparse(class THD *thd); // from sql_yacc.cc /** This is a wrapper of MYSQLparse(). All the code should call parse_sql() instead of MYSQLparse(). As a by product of parsing, the parser can also generate a query digest. To compute a digest, invoke this function as follows. @verbatim THD *thd = ...; const char *query_text = ...; uint query_length = ...; Object_creation_ctx *ctx = ...; bool rc; Parser_state parser_state; if (parser_state.init(thd, query_text, query_length) { ... handle error } parser_state.m_input.m_has_digest= true; parser_state.m_input.m_compute_digest= true; rc= parse_sql(the, &parser_state, ctx); if (! rc) { unsigned char md5[MD5_HASH_SIZE]; char digest_text[1024]; bool truncated; const sql_digest_storage *digest= & thd->m_digest->m_digest_storage; compute_digest_md5(digest, & md5[0]); compute_digest_text(digest, & digest_text[0], sizeof(digest_text), & truncated); } @endverbatim @param thd Thread context. @param parser_state Parser state. @param creation_ctx Object creation context. @return Error status. @retval FALSE on success. @retval TRUE on parsing error. */ bool parse_sql(THD *thd, Parser_state *parser_state, Object_creation_ctx *creation_ctx) { DBUG_ENTER("parse_sql"); bool ret_value; assert(thd->m_parser_state == NULL); // TODO fix to allow parsing gcol exprs after main query. // assert(thd->lex->m_sql_cmd == NULL); MYSQL_QUERY_PARSE_START(const_cast<char*>(thd->query().str)); /* Backup creation context. */ Object_creation_ctx *backup_ctx= NULL; if (creation_ctx) backup_ctx= creation_ctx->set_n_backup(thd); /* Set parser state. */ thd->m_parser_state= parser_state; parser_state->m_digest_psi= NULL; parser_state->m_lip.m_digest= NULL; /* Only consider statements that are supposed to have a digest, like top level queries. */ if (parser_state->m_input.m_has_digest) { /* For these statements, see if the digest computation is required. */ if (thd->m_digest != NULL) { /* Start Digest */ parser_state->m_digest_psi= MYSQL_DIGEST_START(thd->m_statement_psi); if (parser_state->m_input.m_compute_digest || (parser_state->m_digest_psi != NULL)) { /* If either: - the caller wants to compute a digest - the performance schema wants to compute a digest set the digest listener in the lexer. */ parser_state->m_lip.m_digest= thd->m_digest; parser_state->m_lip.m_digest->m_digest_storage.m_charset_number= thd->charset()->number; } } } /* Parse the query. */ /* Use a temporary DA while parsing. We don't know until after parsing whether the current command is a diagnostic statement, in which case we'll need to have the previous DA around to answer questions about it. */ Diagnostics_area *parser_da= thd->get_parser_da(); Diagnostics_area *da= thd->get_stmt_da(); Parser_oom_handler poomh; // Note that we may be called recursively here, on INFORMATION_SCHEMA queries. set_memroot_max_capacity(thd->mem_root, thd->variables.parser_max_mem_size); set_memroot_error_reporting(thd->mem_root, true); thd->push_internal_handler(&poomh); thd->push_diagnostics_area(parser_da, false); bool mysql_parse_status= MYSQLparse(thd) != 0; thd->pop_internal_handler(); set_memroot_max_capacity(thd->mem_root, 0); set_memroot_error_reporting(thd->mem_root, false); /* Unwind diagnostics area. If any issues occurred during parsing, they will become the sole conditions for the current statement. Otherwise, if we have a diagnostic statement on our hands, we'll preserve the previous diagnostics area here so we can answer questions about it. This specifically means that repeatedly asking about a DA won't clear it. Otherwise, it's a regular command with no issues during parsing, so we'll just clear the DA in preparation for the processing of this command. */ if (parser_da->current_statement_cond_count() != 0) { /* Error/warning during parsing: top DA should contain parse error(s)! Any pre-existing conditions will be replaced. The exception is diagnostics statements, in which case we wish to keep the errors so they can be sent to the client. */ if (thd->lex->sql_command != SQLCOM_SHOW_WARNS && thd->lex->sql_command != SQLCOM_GET_DIAGNOSTICS) da->reset_condition_info(thd); /* We need to put any errors in the DA as well as the condition list. */ if (parser_da->is_error() && !da->is_error()) { da->set_error_status(parser_da->mysql_errno(), parser_da->message_text(), parser_da->returned_sqlstate()); } da->copy_sql_conditions_from_da(thd, parser_da); parser_da->reset_diagnostics_area(); parser_da->reset_condition_info(thd); /* Do not clear the condition list when starting execution as it now contains not the results of the previous executions, but a non-zero number of errors/warnings thrown during parsing! */ thd->lex->keep_diagnostics= DA_KEEP_PARSE_ERROR; } thd->pop_diagnostics_area(); /* Check that if MYSQLparse() failed either thd->is_error() is set, or an internal error handler is set. The assert will not catch a situation where parsing fails without an error reported if an error handler exists. The problem is that the error handler might have intercepted the error, so thd->is_error() is not set. However, there is no way to be 100% sure here (the error handler might be for other errors than parsing one). */ assert(!mysql_parse_status || (mysql_parse_status && thd->is_error()) || (mysql_parse_status && thd->get_internal_handler())); /* Reset parser state. */ thd->m_parser_state= NULL; /* Restore creation context. */ if (creation_ctx) creation_ctx->restore_env(thd, backup_ctx); /* That's it. */ ret_value= mysql_parse_status || thd->is_fatal_error; if ((ret_value == 0) && (parser_state->m_digest_psi != NULL)) { /* On parsing success, record the digest in the performance schema. */ assert(thd->m_digest != NULL); MYSQL_DIGEST_END(parser_state->m_digest_psi, & thd->m_digest->m_digest_storage); } MYSQL_QUERY_PARSE_DONE(ret_value); DBUG_RETURN(ret_value); } /** @} (end of group Runtime_Environment) */ /** Check and merge "CHARACTER SET cs [ COLLATE cl ]" clause @param cs character set pointer. @param cl collation pointer. Check if collation "cl" is applicable to character set "cs". If "cl" is NULL (e.g. when COLLATE clause is not specified), then simply "cs" is returned. @return Error status. @retval NULL, if "cl" is not applicable to "cs". @retval pointer to merged CHARSET_INFO on success. */ const CHARSET_INFO* merge_charset_and_collation(const CHARSET_INFO *cs, const CHARSET_INFO *cl) { if (cl) { if (!my_charset_same(cs, cl)) { my_error(ER_COLLATION_CHARSET_MISMATCH, MYF(0), cl->name, cs->csname); return NULL; } return cl; } return cs; }