mysql-server/sql/sql_prepare.cc
2025-03-05 14:31:37 +07:00

3582 lines
120 KiB
C++

/* Copyright (c) 2002, 2024, 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 designed to work 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 either included with
the program or referenced in the documentation.
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 */
/**
@file
This file contains the implementation of prepared statements.
When one prepares a statement:
- Server gets the query from client with command 'COM_STMT_PREPARE';
in the following format:
[COM_STMT_PREPARE:1] [query]
- Parse the query and recognize any parameter markers '?' and
store its information list in lex->param_list
- Allocate a new statement for this prepare; and keep this in
'thd->stmt_map'.
- Without executing the query, return back to client the total
number of parameters along with result-set metadata information
(if any) in the following format:
@verbatim
[STMT_ID:4]
[Column_count:2]
[Param_count:2]
[Params meta info (stubs only for now)] (if Param_count > 0)
[Columns meta info] (if Column_count > 0)
@endverbatim
During prepare the tables used in a statement are opened, but no
locks are acquired. Table opening will block any DDL during the
operation, and we do not need any locks as we neither read nor
modify any data during prepare. Tables are closed after prepare
finishes.
When one executes a statement:
- Server gets the command 'COM_STMT_EXECUTE' to execute the
previously prepared query. If there are any parameter markers, then the
client will send the data in the following format:
@verbatim
[COM_STMT_EXECUTE:1]
[STMT_ID:4]
[NULL_BITS:(param_count+7)/8)]
[TYPES_SUPPLIED_BY_CLIENT(0/1):1]
[[length]data]
[[length]data] .. [[length]data].
@endverbatim
(Note: Except for string/binary types; all other types will not be
supplied with length field)
- If it is a first execute or types of parameters were altered by client,
then setup the conversion routines.
- Assign parameter items from the supplied data.
- Execute the query without re-parsing and send back the results
to client
During execution of prepared statement tables are opened and locked
the same way they would for normal (non-prepared) statement
execution. Tables are unlocked and closed after the execution.
When one supplies long data for a placeholder:
- Server gets the long data in pieces with command type
'COM_STMT_SEND_LONG_DATA'.
- The packet received will have the format as:
[COM_STMT_SEND_LONG_DATA:1][STMT_ID:4][parameter_number:2][data]
- data from the packet is appended to the long data value buffer for this
placeholder.
- It's up to the client to stop supplying data chunks at any point. The
server doesn't care; also, the server doesn't notify the client whether
it got the data or not; if there is any error, then it will be returned
at statement execute.
*/
#include "sql/sql_prepare.h"
#include "my_config.h"
#include <limits.h>
#include <stdint.h>
#include <stdio.h>
#include <string.h>
#include <algorithm>
#include <limits>
#include <memory>
#include <unordered_map>
#include <utility>
#include "decimal.h"
#include "field_types.h"
#include "map_helpers.h"
#include "my_alloc.h"
#include "my_byteorder.h"
#include "my_command.h"
#include "my_compiler.h"
#include "my_dbug.h"
#include "my_sqlcommand.h"
#include "my_sys.h"
#include "my_time.h"
#include "mysql/com_data.h"
#include "mysql/components/services/log_shared.h"
#include "mysql/plugin_audit.h"
#include "mysql/psi/mysql_mutex.h"
#include "mysql/psi/mysql_ps.h" // MYSQL_EXECUTE_PS
#include "mysql/strings/dtoa.h"
#include "mysql/strings/int2str.h"
#include "mysql/strings/m_ctype.h"
#include "mysql/udf_registration_types.h"
#include "mysql_com.h"
#include "mysql_time.h"
#include "mysqld_error.h"
#include "nulls.h"
#include "scope_guard.h"
#include "sql-common/my_decimal.h"
#include "sql/auth/auth_acls.h"
#include "sql/auth/auth_common.h" // check_table_access
#include "sql/auth/sql_security_ctx.h"
#include "sql/binlog.h"
#include "sql/debug_sync.h"
#include "sql/derror.h" // ER_THD
#include "sql/handler.h"
#include "sql/item.h"
#include "sql/item_func.h" // user_var_entry
#include "sql/log.h" // query_logger
#include "sql/mdl.h"
#include "sql/mysqld.h" // opt_general_log
#include "sql/opt_trace.h" // Opt_trace_array
#include "sql/protocol.h"
#include "sql/protocol_classic.h"
#include "sql/psi_memory_key.h"
#include "sql/query_options.h"
#include "sql/query_result.h"
#include "sql/resourcegroups/resource_group_basic_types.h"
#include "sql/resourcegroups/resource_group_mgr.h"
#include "sql/session_tracker.h"
#include "sql/set_var.h" // set_var_base
#include "sql/sp_cache.h" // sp_cache_enforce_limit
#include "sql/sql_audit.h" // mysql_global_audit_mask
#include "sql/sql_base.h" // open_tables_for_query, open_temporary_table
#include "sql/sql_class.h"
#include "sql/sql_cmd.h"
#include "sql/sql_cmd_ddl_table.h"
#include "sql/sql_const.h"
#include "sql/sql_cursor.h" // Server_side_cursor
#include "sql/sql_db.h" // mysql_change_db
#include "sql/sql_digest_stream.h"
#include "sql/sql_handler.h" // mysql_ha_rm_tables
#include "sql/sql_insert.h" // Query_result_create
#include "sql/sql_lex.h"
#include "sql/sql_parse.h" // sql_command_flags
#include "sql/sql_profile.h"
#include "sql/sql_query_rewrite.h"
#include "sql/sql_rewrite.h" // mysql_rewrite_query
#include "sql/sql_view.h" // create_view_precheck
#include "sql/statement/statement_runnable.h"
#include "sql/system_variables.h"
#include "sql/table.h"
#include "sql/thd_raii.h"
#include "sql/thr_malloc.h"
#include "sql/transaction.h" // trans_rollback_implicit
#include "sql/window.h"
#include "sql_string.h"
#include "string_with_len.h"
#include "violite.h"
namespace resourcegroups {
class Resource_group;
} // namespace resourcegroups
using std::max;
using std::min;
/****************************************************************************/
namespace {
/**
A result class used to send cursor rows using the binary protocol.
*/
class Query_fetch_protocol_binary final : public Query_result_send {
Protocol_binary protocol;
public:
explicit Query_fetch_protocol_binary(THD *thd)
: Query_result_send(), protocol(thd) {}
bool send_result_set_metadata(THD *thd, const mem_root_deque<Item *> &list,
uint flags) override;
bool send_data(THD *thd, const mem_root_deque<Item *> &items) override;
bool send_eof(THD *thd) override;
bool use_protocol_adapter() const override { return false; }
};
} // namespace
/******************************************************************************
Implementation
******************************************************************************/
/**
Rewrite the current query (to obfuscate passwords etc.).
Side-effect: thd->rewritten_query() may be populated with a rewritten
query. If the query is not of a rewritable type,
thd->rewritten_query() will be empty.
@param thd thread handle
*/
void rewrite_query(THD *thd) {
/*
thd->m_rewritten_query may already contain "PREPARE stmt FROM ..."
at this point, so we reset it here so mysql_rewrite_query()
won't complain.
*/
thd->reset_rewritten_query();
/*
Now replace the "PREPARE ..." with the obfuscated version of the
actual query were prepare.
*/
mysql_rewrite_query(thd);
}
/**
Unless we're doing dynamic SQL, write the current query to the
general query log if it's open. If we have a rewritten version
of the query, use that instead of the "raw" one.
Side-effect: query may be written to general log if it's open.
@param thd thread handle
*/
void log_execute_line(THD *thd) {
/*
Do not print anything if this is an SQL prepared statement and
we're inside a stored procedure (also called Dynamic SQL) --
sub-statements inside stored procedures are not logged into
the general log.
*/
if (thd->sp_runtime_ctx != nullptr) return;
if (thd->rewritten_query().length())
query_logger.general_log_write(thd, COM_STMT_EXECUTE,
thd->rewritten_query().ptr(),
thd->rewritten_query().length());
else
query_logger.general_log_write(thd, COM_STMT_EXECUTE, thd->query().str,
thd->query().length);
}
class Statement_backup {
LEX *m_lex;
LEX_CSTRING m_query_string;
String m_rewritten_query;
bool m_safe_to_display;
public:
LEX *lex() const { return m_lex; }
/**
Prepared the THD to execute the prepared statement.
Save the current THD statement state.
*/
void set_thd_to_ps(THD *thd, Prepared_statement *stmt) {
DBUG_TRACE;
mysql_mutex_lock(&thd->LOCK_thd_data);
m_lex = thd->lex;
thd->lex = stmt->m_lex;
mysql_mutex_unlock(&thd->LOCK_thd_data);
m_query_string = thd->query();
thd->set_query(stmt->m_query_string);
m_safe_to_display = thd->safe_to_display();
/* Keep the current behaviour of displaying prepared statements always by
default. This can be changed in future if required. */
thd->set_safe_display(true);
return;
}
/**
Restore the THD statement state after the prepared
statement has finished executing.
*/
void restore_thd(THD *thd, Prepared_statement *stmt) {
DBUG_TRACE;
mysql_mutex_lock(&thd->LOCK_thd_data);
stmt->m_lex = thd->lex;
thd->lex = m_lex;
mysql_mutex_unlock(&thd->LOCK_thd_data);
thd->set_safe_display(m_safe_to_display);
stmt->m_query_string = thd->query();
thd->set_query(m_query_string);
}
/**
Save the current rewritten query prior to
rewriting the prepared statement.
*/
void save_rlb(THD *thd) {
DBUG_TRACE;
if (thd->rewritten_query().length() > 0) {
/* Duplicate the original rewritten query. */
m_rewritten_query.copy(thd->rewritten_query());
/* Swap the duplicate with the original. */
thd->swap_rewritten_query(m_rewritten_query);
}
return;
}
/**
Restore the rewritten query after the prepared
statement has finished executing.
*/
void restore_rlb(THD *thd) {
DBUG_TRACE;
if (m_rewritten_query.length() > 0) {
/* Restore with swap() instead of '='. */
thd->swap_rewritten_query(m_rewritten_query);
/* Free the rewritten prepared statement. */
m_rewritten_query.mem_free();
}
return;
}
};
/**
Set source parameter type based on information from protocol buffer
@param param Parameter to set source type for
@param type Field type
@param unsigned_type true if type is unsigned (applicable for numeric types)
@param cs_source The source collation for non-binary string parameters
If parameter metadata is invalid, set source data type MYSQL_TYPE_INVALID.
*/
static void set_parameter_type(Item_param *param, enum enum_field_types type,
bool unsigned_type,
const CHARSET_INFO *cs_source) {
param->set_data_type_source(type, unsigned_type);
switch (param->data_type_source()) {
case MYSQL_TYPE_NULL:
case MYSQL_TYPE_TINY:
case MYSQL_TYPE_SHORT:
case MYSQL_TYPE_INT24:
case MYSQL_TYPE_LONG:
case MYSQL_TYPE_LONGLONG:
case MYSQL_TYPE_FLOAT:
case MYSQL_TYPE_DOUBLE:
case MYSQL_TYPE_DECIMAL:
case MYSQL_TYPE_NEWDECIMAL:
case MYSQL_TYPE_DATE:
case MYSQL_TYPE_TIME:
case MYSQL_TYPE_DATETIME:
case MYSQL_TYPE_TIMESTAMP:
break;
case MYSQL_TYPE_TINY_BLOB:
case MYSQL_TYPE_MEDIUM_BLOB:
case MYSQL_TYPE_LONG_BLOB:
case MYSQL_TYPE_BLOB: {
param->set_collation_source(&my_charset_bin);
break;
}
case MYSQL_TYPE_VARCHAR:
case MYSQL_TYPE_JSON:
case MYSQL_TYPE_VAR_STRING:
case MYSQL_TYPE_STRING: {
param->set_collation_source(cs_source);
break;
}
default:
param->set_data_type_source(MYSQL_TYPE_INVALID, false);
break;
}
}
/**
Set parameter value from protocol buffer
@param param Parameter to set a value for
@param pos Pointer to parameter value in protocol buffer
@param len Length of parameter value
@returns false on success, true on error
The function reads a binary valuefrom pos, converts it to the requested type
and assigns it to the paramameter.
*/
static bool set_parameter_value(
Item_param *param, const uchar **pos, ulong len,
enum Prepared_statement::enum_param_pack_type pack_type) {
switch (param->data_type_source()) {
case MYSQL_TYPE_TINY: {
assert(len >= 1);
const int8 value = (int8) * *pos;
if (param->is_unsigned_actual())
param->set_int((ulonglong)((uint8)value));
else
param->set_int((longlong)value);
break;
}
case MYSQL_TYPE_SHORT: {
assert(len >= 2);
const int16 value = sint2korr(*pos);
if (param->is_unsigned_actual())
param->set_int((ulonglong)((uint16)value));
else
param->set_int((longlong)value);
break;
}
case MYSQL_TYPE_LONG: {
assert(len >= 4);
const int32 value = sint4korr(*pos);
if (param->is_unsigned_actual())
param->set_int((ulonglong)((uint32)value));
else
param->set_int((longlong)value);
break;
}
case MYSQL_TYPE_LONGLONG: {
assert(len >= 8);
const longlong value = sint8korr(*pos);
if (param->is_unsigned_actual())
param->set_int((ulonglong)value);
else
param->set_int(value);
break;
}
case MYSQL_TYPE_FLOAT: {
assert(len >= 4);
const float data = float4get(*pos);
param->set_double((double)data);
break;
}
case MYSQL_TYPE_DOUBLE: {
assert(len >= 8);
const double data = float8get(*pos);
param->set_double(data);
break;
}
case MYSQL_TYPE_DECIMAL:
case MYSQL_TYPE_NEWDECIMAL: {
param->set_decimal((const char *)*pos, len);
break;
}
case MYSQL_TYPE_TIME: {
MYSQL_TIME tm;
if (pack_type == Prepared_statement::enum_param_pack_type::UNPACKED) {
assert(len == sizeof(MYSQL_TIME));
tm = *(*(const MYSQL_TIME **)pos);
} else {
if (len >= 8) {
const uchar *to = *pos;
tm.neg = (bool)to[0];
const uint day = (uint)sint4korr(to + 1);
tm.hour = (uint)to[5] + day * 24;
tm.minute = (uint)to[6];
tm.second = (uint)to[7];
tm.second_part = (len > 8) ? (ulong)sint4korr(to + 8) : 0;
if (tm.hour > 838) {
/* TODO: add warning 'Data truncated' here */
tm.hour = 838;
tm.minute = 59;
tm.second = 59;
}
tm.day = tm.year = tm.month = 0;
} else {
set_zero_time(&tm, MYSQL_TIMESTAMP_TIME);
}
}
param->set_time(&tm, MYSQL_TIMESTAMP_TIME);
break;
}
case MYSQL_TYPE_DATE: {
MYSQL_TIME tm;
if (pack_type == Prepared_statement::enum_param_pack_type::UNPACKED) {
assert(len == sizeof(MYSQL_TIME));
tm = *(*(const MYSQL_TIME **)pos);
} else {
if (len >= 4) {
const uchar *to = *pos;
tm.year = (uint)sint2korr(to);
tm.month = (uint)to[2];
tm.day = (uint)to[3];
tm.hour = tm.minute = tm.second = 0;
tm.second_part = 0;
tm.neg = false;
} else {
set_zero_time(&tm, MYSQL_TIMESTAMP_DATE);
}
}
param->set_time(&tm, MYSQL_TIMESTAMP_DATE);
break;
}
case MYSQL_TYPE_DATETIME:
case MYSQL_TYPE_TIMESTAMP: {
MYSQL_TIME tm;
enum_mysql_timestamp_type type = MYSQL_TIMESTAMP_DATETIME;
if (pack_type == Prepared_statement::enum_param_pack_type::UNPACKED) {
assert(len == sizeof(MYSQL_TIME));
tm = *(*(const MYSQL_TIME **)pos);
} else {
assert(len == 0 || len == 4 || len == 7 || len == 11 || len == 13);
const uchar *to = *pos;
if (len < 4) {
set_zero_time(&tm, MYSQL_TIMESTAMP_DATETIME);
} else {
tm.neg = false;
tm.year = (uint)sint2korr(to);
tm.month = (uint)to[2];
tm.day = (uint)to[3];
}
if (len >= 7) {
tm.hour = (uint)to[4];
tm.minute = (uint)to[5];
tm.second = (uint)to[6];
} else { // len == 4
tm.hour = tm.minute = tm.second = 0;
}
tm.second_part =
(len >= 11) ? static_cast<std::uint64_t>(sint4korr(to + 7)) : 0;
if (len >= 13) {
tm.time_zone_displacement = sint2korr(to + 11) * SECS_PER_MIN;
type = MYSQL_TIMESTAMP_DATETIME_TZ;
}
}
param->set_time(&tm, type);
break;
}
case MYSQL_TYPE_TINY_BLOB:
case MYSQL_TYPE_MEDIUM_BLOB:
case MYSQL_TYPE_LONG_BLOB:
case MYSQL_TYPE_BLOB: {
param->set_str((const char *)*pos, len);
break;
}
case MYSQL_TYPE_VARCHAR:
case MYSQL_TYPE_JSON:
case MYSQL_TYPE_VAR_STRING:
case MYSQL_TYPE_STRING: {
param->set_str((const char *)*pos, len);
break;
}
default:
/*
The client library ensures that only the type codes listed above
are encountered. The 'default' label makes it possible to handle
malformed packets as well.
*/
break;
}
return false;
}
/**
Check whether this parameter data type is compatible with long data.
Used to detect whether a long data stream has been supplied to a
incompatible data type.
Notice that VARCHAR type is MYSQL_TYPE_STRING, not MYSQL_TYPE_VARCHAR,
when sent on protocol between client and server.
*/
inline bool is_param_long_data_type(Item_param *param) {
return param->data_type_source() >= MYSQL_TYPE_TINY_BLOB &&
param->data_type_source() <= MYSQL_TYPE_STRING;
}
/**
Assign parameter values from data supplied by the client.
If required, generate a valid non-parameterized query for logging.
@param thd current thread.
@param query The query with parameter markers replaced with values
supplied by user that were used to execute the query.
@param has_new_types if true, new types of actual parameters are provided,
otherwise use the parameters from previous execution.
@param parameters Array of actual parameter values.
Contains parameter types if has_new_types is true.
@param pack_type UNPACKED means that the parameter value buffer points to
MYSQL_TIME*
@returns false if success, true if error
@note
m_with_log is set when one of slow or general logs are open.
Logging of prepared statements in all cases is performed
by means of regular queries: if parameter data was supplied from C API,
each placeholder in the query is replaced with its actual value;
if we're logging a [Dynamic] SQL prepared statement, parameter markers
are replaced with variable names.
Example:
@verbatim
mysqld_stmt_prepare("UPDATE t1 SET a=a*1.25 WHERE a=?")
--> general logs gets [Prepare] UPDATE t1 SET a*1.25 WHERE a=?"
mysqld_stmt_execute(stmt);
--> general and binary logs get
[Execute] UPDATE t1 SET a*1.25 WHERE a=1"
@endverbatim
If a statement has been prepared using SQL syntax:
@verbatim
PREPARE stmt FROM "UPDATE t1 SET a=a*1.25 WHERE a=?"
--> general log gets [Query] PREPARE stmt FROM "UPDATE ..."
EXECUTE stmt USING @a
--> general log gets [Query] EXECUTE stmt USING @a;
@endverbatim
*/
bool Prepared_statement::insert_parameters(
THD *thd, String *query, bool has_new_types, PS_PARAM *parameters,
enum enum_param_pack_type pack_type) {
DBUG_TRACE;
Item_param **end = m_param_array + m_param_count;
size_t length = 0;
// Reserve an extra space of 32 bytes for each placeholder parameter.
if (m_with_log && query->reserve(m_query_string.length + 32 * m_param_count))
return true;
uint i = 0;
for (Item_param **it = m_param_array; it < end; ++it, i++) {
Item_param *const param = *it;
if (has_new_types) {
set_parameter_type(param, parameters[i].type, parameters[i].unsigned_type,
thd->variables.character_set_client);
// Client may have provided invalid metadata:
if (param->data_type_source() == MYSQL_TYPE_INVALID) {
my_error(ER_WRONG_ARGUMENTS, MYF(0), "mysqld_stmt_execute");
return true;
}
}
if (param->param_state() == Item_param::LONG_DATA_VALUE) {
/*
A long data stream was supplied for this parameter marker.
This was done after prepare, prior to providing a placeholder
type (the types are supplied at execute). Check that the
supplied type of placeholder can accept a data stream.
*/
if (!is_param_long_data_type(param)) {
my_error(ER_WRONG_ARGUMENTS, MYF(0), "mysqld_stmt_execute");
return true;
}
param->set_data_type_actual(MYSQL_TYPE_VARCHAR);
// see Item_param::set_str() for explanation
param->set_collation_actual(
param->collation_source() == &my_charset_bin ? &my_charset_bin
: param->collation.collation != &my_charset_bin
? param->collation.collation
: current_thd->variables.collation_connection);
} else if (parameters[i].null_bit) {
param->set_null();
} else {
if (set_parameter_value(param, &parameters[i].value, parameters[i].length,
pack_type)) {
return true;
}
// NO_VALUE probably means broken client, no metadata provided.
if (param->param_state() == Item_param::NO_VALUE) {
my_error(ER_WRONG_ARGUMENTS, MYF(0), "mysqld_stmt_execute");
return true;
}
/*
Pinning of data types is only implemented for numeric types
(integers, decimal values and float values). String values with numeric
data is also accepted as parameter values.
*/
assert(!param->is_type_pinned() || param->result_type() == INT_RESULT ||
param->result_type() == DECIMAL_RESULT ||
param->result_type() == REAL_RESULT);
if (param->is_type_pinned()) {
// Accept string values from client
if (param->data_type() == MYSQL_TYPE_LONGLONG) {
if (param->data_type_actual() == MYSQL_TYPE_VARCHAR) {
const longlong val = param->val_int();
if (thd->is_error()) return true;
if (param->unsigned_flag)
param->set_int((ulonglong)val);
else
param->set_int(val);
} else if (param->data_type_actual() != MYSQL_TYPE_LONGLONG) {
my_error(ER_WRONG_ARGUMENTS, MYF(0), "mysqld_stmt_execute");
return true;
}
if ((param->unsigned_flag && !param->is_unsigned_actual() &&
param->value.integer < 0) ||
(!param->unsigned_flag && param->is_unsigned_actual() &&
param->value.integer < 0)) {
my_error(ER_DATA_OUT_OF_RANGE, MYF(0), "signed integer",
"mysqld_stmt_execute");
return true;
}
} else if (param->data_type() == MYSQL_TYPE_NEWDECIMAL) {
if (param->data_type_actual() == MYSQL_TYPE_VARCHAR ||
param->data_type_actual() == MYSQL_TYPE_LONGLONG) {
my_decimal val;
my_decimal *v = param->val_decimal(&val);
if (thd->is_error()) return true;
param->set_decimal(v);
} else if (param->data_type_actual() != MYSQL_TYPE_NEWDECIMAL) {
my_error(ER_WRONG_ARGUMENTS, MYF(0), "mysqld_stmt_execute");
return true;
}
} else if (param->data_type() == MYSQL_TYPE_DOUBLE) {
if (param->data_type_actual() == MYSQL_TYPE_VARCHAR ||
param->data_type_actual() == MYSQL_TYPE_LONGLONG ||
param->data_type_actual() == MYSQL_TYPE_NEWDECIMAL) {
const double val = param->val_real();
if (thd->is_error()) return true;
param->set_double(val);
} else if (param->data_type_actual() != MYSQL_TYPE_DOUBLE) {
my_error(ER_WRONG_ARGUMENTS, MYF(0), "mysqld_stmt_execute");
return true;
}
}
}
}
if (m_with_log) {
String str;
const String *val = param->query_val_str(thd, &str);
if (val == nullptr) return true;
if (param->convert_value()) return true; /* out of memory */
const size_t num_bytes = param->pos_in_query - length;
if (query->length() + num_bytes + val->length() >
std::numeric_limits<uint32>::max()) {
my_error(ER_WRONG_ARGUMENTS, MYF(0), "mysqld_stmt_execute");
return true;
}
if (query->append(m_query_string.str + length, num_bytes) ||
query->append(*val))
return true;
length = param->pos_in_query + 1;
} else {
if (param->convert_value()) return true; /* out of memory */
}
param->sync_clones();
}
// Copy part of query string after last parameter marker
if (m_with_log && query->append(m_query_string.str + length,
m_query_string.length - length))
return true;
return false;
}
/**
Copy parameter metada data from parameter array into current prepared stmt.
@param from_param_array Parameter array to copy from.
*/
void Prepared_statement::copy_parameter_types(Item_param **from_param_array) {
for (uint i = 0; i < m_param_count; ++i) {
Item_param *from = from_param_array[i];
Item_param *to = this->m_param_array[i];
to->copy_param_actual_type(from);
}
}
/**
Setup data conversion routines using an array of parameter
markers from the original prepared statement.
Swap the parameter data of the original prepared
statement to the new one.
Used only when we re-prepare a prepared statement.
There are two reasons for this function to exist:
1) In the binary client/server protocol, parameter metadata
is sent only at first execute. Consequently, if we need to
reprepare a prepared statement at a subsequent execution,
we may not have metadata information in the packet.
In that case we use the parameter array of the original
prepared statement to setup parameter types of the new
prepared statement.
2) In the binary client/server protocol, we may supply
long data in pieces. When the last piece is supplied,
we assemble the pieces and convert them from client
character set to the connection character set. After
that the parameter value is only available inside
the parameter, the original pieces are lost, and thus
we can only assign the corresponding parameter of the
reprepared statement from the original value.
@param[out] param_array_dst parameter markers of the new statement
@param[in] param_array_src parameter markers of the original
statement
@param[in] param_count total number of parameters. Is the
same in src and dst arrays, since
the statement query is the same
*/
static void swap_parameter_array(Item_param **param_array_dst,
Item_param **param_array_src,
uint param_count) {
Item_param **dst = param_array_dst;
Item_param **src = param_array_src;
Item_param **end = param_array_dst + param_count;
for (; dst < end; ++src, ++dst) {
(*dst)->set_param_type_and_swap_value(*src);
(*dst)->sync_clones();
(*src)->sync_clones();
}
}
/**
Assign prepared statement parameters from user variables.
If m_with_log is set, also construct query string for binary log.
@param thd Current thread.
@param varnames List of variables. Caller must ensure that number
of variables in the list is equal to number of statement
parameters
@param query The query with parameter markers replaced with corresponding
user variables that were used to execute the query.
@returns false if success, true if error
*/
bool Prepared_statement::insert_parameters_from_vars(THD *thd,
List<LEX_STRING> &varnames,
String *query) {
Item_param **end = m_param_array + m_param_count;
List_iterator<LEX_STRING> var_it(varnames);
StringBuffer<STRING_BUFFER_USUAL_SIZE> buf;
size_t length = 0;
DBUG_TRACE;
// Reserve an extra space of 32 bytes for each placeholder parameter.
if (m_with_log) query->reserve(m_query_string.length + 32 * m_param_count);
// Protects thd->user_vars
mysql_mutex_lock(&thd->LOCK_thd_data);
for (Item_param **it = m_param_array; it < end; ++it) {
Item_param *const param = *it;
LEX_STRING *const varname = var_it++;
user_var_entry *entry =
find_or_nullptr(thd->user_vars, to_string(*varname));
if (m_with_log) {
if (entry != nullptr) {
set_parameter_type(param, Item::result_to_type(entry->type()),
entry->unsigned_flag, entry->collation.collation);
} else {
set_parameter_type(param, param->data_type(), param->unsigned_flag,
param->collation.collation);
}
if (param->set_from_user_var(thd, entry)) goto error;
const String *val = param->query_val_str(thd, &buf);
if (val == nullptr) goto error;
if (param->convert_value()) goto error;
const size_t num_bytes = param->pos_in_query - length;
if (query->length() + num_bytes + val->length() >
std::numeric_limits<uint32>::max())
goto error;
if (query->append(m_query_string.str + length, num_bytes) ||
query->append(*val))
goto error;
length = param->pos_in_query + 1;
} else {
assert(false);
if (param->set_from_user_var(thd, entry)) goto error;
if (entry) length += entry->length();
if (length > std::numeric_limits<uint32>::max() || param->convert_value())
goto error;
}
param->sync_clones();
}
// Copy part of query string after last parameter marker
if (m_with_log && query->append(m_query_string.str + length,
m_query_string.length - length))
goto error;
mysql_mutex_unlock(&thd->LOCK_thd_data);
return false;
error:
mysql_mutex_unlock(&thd->LOCK_thd_data);
return true;
}
static bool send_statement(THD *thd, const Prepared_statement *stmt,
uint no_columns, Query_result *result,
const mem_root_deque<Item *> *types) {
// Send the statement status(id, no_columns, no_params, error_count);
bool rc = thd->get_protocol()->store_ps_status(
stmt->id(), no_columns, stmt->m_param_count,
thd->get_stmt_da()->current_statement_cond_count());
if (!rc && stmt->m_param_count != 0) {
// Send the list of parameters
mem_root_deque<Item *> param_list(thd->mem_root);
for (Item_param &item : stmt->m_lex->param_list) {
param_list.push_back(&item);
}
rc |= thd->send_result_metadata(param_list, Protocol::SEND_EOF);
assert(CountVisibleFields(param_list) == stmt->m_param_count);
}
if (rc) return true; /* purecov: inspected */
// Send
if (types && result &&
result->send_result_set_metadata(thd, *types, Protocol::SEND_EOF))
return true; /* purecov: inspected */
// Flag that a response has already been sent
thd->get_stmt_da()->disable_status();
// Flush the result only if previous statements succeeded.
if (!rc) {
thd->get_stmt_da()->set_overwrite_status(true);
rc |= thd->get_protocol()->flush();
thd->get_stmt_da()->set_overwrite_status(false);
}
return rc;
}
/**
Validate and prepare for execution SET statement expressions.
@param thd current thread
@param stmt prepared statement
@param tables list of tables used in this query
@param var_list list of expressions
@returns false if success, true if error
*/
static bool mysql_test_set_fields(THD *thd,
Prepared_statement *stmt [[maybe_unused]],
Table_ref *tables,
List<set_var_base> *var_list) {
List_iterator_fast<set_var_base> it(*var_list);
set_var_base *var;
DBUG_TRACE;
assert(stmt->m_arena.is_stmt_prepare());
if (tables &&
check_table_access(thd, SELECT_ACL, tables, false, UINT_MAX, false))
return true; /* purecov: inspected */
if (open_tables_for_query(thd, tables, MYSQL_OPEN_FORCE_SHARED_MDL))
return true; /* purecov: inspected */
thd->lex->set_using_hypergraph_optimizer(
thd->optimizer_switch_flag(OPTIMIZER_SWITCH_HYPERGRAPH_OPTIMIZER));
const Prepared_stmt_arena_holder ps_arena_holder(thd);
while ((var = it++)) {
if (var->light_check(thd)) return true; /* purecov: inspected */
var->cleanup();
}
thd->lex->unit->set_prepared();
thd->lex->save_cmd_properties(thd);
thd->lex->cleanup(false);
return false;
}
/**
@param thd thread handler
@param result query result sink
@param added_options options added to query block before preparation
@returns false if success, true if error
*/
static bool select_like_stmt_test(THD *thd, Query_result *result,
ulonglong added_options) {
DBUG_TRACE;
LEX *const lex = thd->lex;
lex->query_block->context.resolve_in_select_list = true;
if (lex->unit->prepare(thd, result, nullptr, added_options, 0)) {
return true;
}
lex->save_cmd_properties(thd);
return false;
}
/**
Validate and prepare for execution CREATE TABLE statement.
@param thd Thread handle.
@returns false if success, true if error
*/
bool Sql_cmd_create_table::prepare(THD *thd) {
LEX *const lex = thd->lex;
Query_block *query_block = lex->query_block;
Table_ref *create_table = lex->query_tables;
DBUG_TRACE;
if (create_table_precheck(thd, query_expression_tables, create_table))
return true;
auto cleanup_se_guard = create_scope_guard(
[lex] { lex->set_secondary_engine_execution_context(nullptr); });
if (!query_block->fields.empty()) {
/* Base table and temporary table are not in the same name space. */
if (!(lex->create_info->options & HA_LEX_CREATE_TMP_TABLE))
create_table->open_type = OT_BASE_ONLY;
if (open_tables_for_query(thd, lex->query_tables,
MYSQL_OPEN_FORCE_SHARED_MDL))
return true;
query_block->context.resolve_in_select_list = true;
lex->set_using_hypergraph_optimizer(
thd->optimizer_switch_flag(OPTIMIZER_SWITCH_HYPERGRAPH_OPTIMIZER));
const Prepared_stmt_arena_holder ps_arena_holder(thd);
Query_result *result = new (thd->mem_root)
Query_result_create(create_table, &query_block->fields, lex->duplicates,
query_expression_tables);
if (result == nullptr) return true;
bool link_to_local;
lex->unlink_first_table(&link_to_local);
const bool res = select_like_stmt_test(thd, result, SELECT_NO_UNLOCK);
lex->link_first_table_back(create_table, link_to_local);
if (res) return true;
} else {
/*
Check that the source table exist, and also record
its metadata version. Even though not strictly necessary,
we validate metadata of all CREATE TABLE statements,
which keeps metadata validation code simple.
*/
if (open_tables_for_query(thd, lex->query_tables,
MYSQL_OPEN_FORCE_SHARED_MDL))
return true;
lex->set_using_hypergraph_optimizer(
thd->optimizer_switch_flag(OPTIMIZER_SWITCH_HYPERGRAPH_OPTIMIZER));
}
set_prepared();
return false;
}
/**
@brief Validate and prepare for execution CREATE VIEW statement
@param thd current thread
@param stmt prepared statement
@note This function handles create view commands.
@returns false if success, true if error
@retval false Operation was a success.
@retval true An error occurred.
*/
static bool mysql_test_create_view(THD *thd, Prepared_statement *stmt) {
DBUG_TRACE;
assert(stmt->m_arena.is_stmt_prepare());
LEX *lex = stmt->m_lex;
Query_block *const select = lex->query_block;
bool res = true;
/* Skip first table, which is the view we are creating */
bool link_to_local;
Table_ref *view = lex->unlink_first_table(&link_to_local);
Table_ref *tables = lex->query_tables;
if (create_view_precheck(thd, tables, view, lex->create_view_mode)) goto err;
/*
Since we can't pre-open temporary tables for SQLCOM_CREATE_VIEW,
(see mysql_create_view) we have to do it here instead.
*/
if (open_temporary_tables(thd, tables)) goto err;
if (open_tables_for_query(thd, tables, MYSQL_OPEN_FORCE_SHARED_MDL)) goto err;
lex->context_analysis_only |= CONTEXT_ANALYSIS_ONLY_VIEW;
{
const Prepared_stmt_arena_holder ps_arena_holder(thd);
res = select_like_stmt_test(thd, nullptr, 0);
if (res) goto err;
/* Check if the auto generated column names are conforming. */
make_valid_column_names(lex);
/*
Only column names of the first query block should be checked for
duplication; any further UNION-ed part isn't used for determining
names of the view's columns.
*/
if (check_duplicate_names(view->derived_column_names(), select->fields,
true)) {
res = true;
goto err;
}
/*
Make sure the view doesn't have so many columns that we hit the
64k header limit if the view is materialized as a MyISAM table.
*/
if (select->fields.size() > MAX_FIELDS) {
my_error(ER_TOO_MANY_FIELDS, MYF(0));
res = true;
goto err;
}
}
err:
/* put view back for PS rexecuting */
lex->link_first_table_back(view, link_to_local);
return res;
}
/**
Perform semantic analysis of query and send a response packet to client.
This function
- opens all tables and checks privileges to them.
- validates semantics of statement columns and SQL functions
by calling fix_fields.
- performs global transformations such as semi-join conversion.
@param thd current thread
@retval false success, statement metadata is sent to client
@retval true error, error message is set in THD (but not sent)
*/
bool Prepared_statement::prepare_query(THD *thd) {
DBUG_TRACE;
assert(m_lex == thd->lex); // set_n_backup_active_arena() guarantees that
Query_block *query_block = m_lex->query_block;
enum enum_sql_command sql_command = m_lex->sql_command;
int res = 0;
DBUG_PRINT("enter",
("command: %d param_count: %u", sql_command, m_param_count));
m_lex->first_lists_tables_same();
Table_ref *const tables = m_lex->query_tables;
/* set context for commands which do not use setup_tables */
query_block->context.resolve_in_table_list_only(
query_block->get_table_list());
/*
For the optimizer trace, this is the symmetric, for statement preparation,
of what is done at statement execution (in mysql_execute_command()).
*/
Opt_trace_start ots(thd, tables, sql_command, &m_lex->var_list,
thd->query().str, thd->query().length, nullptr,
thd->variables.character_set_client);
const Opt_trace_object trace_command(&thd->opt_trace);
const Opt_trace_array trace_command_steps(&thd->opt_trace, "steps");
if ((m_lex->keep_diagnostics == DA_KEEP_COUNTS) ||
(m_lex->keep_diagnostics == DA_KEEP_DIAGNOSTICS)) {
my_error(ER_UNSUPPORTED_PS, MYF(0));
return true;
}
if (sql_command_flags[sql_command] & CF_HA_CLOSE)
mysql_ha_rm_tables(thd, tables);
/*
Open temporary tables that are known now. Temporary tables added by
prelocking will be opened afterwards (during open_tables()).
*/
if (sql_command_flags[sql_command] & CF_PREOPEN_TMP_TABLES) {
if (open_temporary_tables(thd, tables)) return true;
}
switch (sql_command) {
case SQLCOM_CREATE_VIEW:
if (m_lex->create_view_mode == enum_view_create_mode::VIEW_ALTER) {
my_error(ER_UNSUPPORTED_PS, MYF(0));
return true;
}
res = mysql_test_create_view(thd, this);
break;
case SQLCOM_SET_PASSWORD:
case SQLCOM_SET_OPTION:
res = mysql_test_set_fields(thd, this, tables, &m_lex->var_list);
break;
/*
Note that we don't need to have cases in this list if they are
marked with CF_STATUS_COMMAND in sql_command_flags
*/
case SQLCOM_DROP_TABLE:
case SQLCOM_RENAME_TABLE:
case SQLCOM_ALTER_TABLE:
case SQLCOM_COMMIT:
case SQLCOM_CREATE_INDEX:
case SQLCOM_DROP_INDEX:
case SQLCOM_ROLLBACK:
case SQLCOM_TRUNCATE:
case SQLCOM_DROP_VIEW:
case SQLCOM_REPAIR:
case SQLCOM_ANALYZE:
case SQLCOM_OPTIMIZE:
case SQLCOM_CHANGE_REPLICATION_SOURCE:
case SQLCOM_CHANGE_REPLICATION_FILTER:
case SQLCOM_RESET:
case SQLCOM_FLUSH:
case SQLCOM_REPLICA_START:
case SQLCOM_REPLICA_STOP:
case SQLCOM_INSTALL_PLUGIN:
case SQLCOM_UNINSTALL_PLUGIN:
case SQLCOM_CREATE_DB:
case SQLCOM_DROP_DB:
case SQLCOM_CHECKSUM:
case SQLCOM_CREATE_USER:
case SQLCOM_RENAME_USER:
case SQLCOM_DROP_USER:
case SQLCOM_ALTER_USER:
case SQLCOM_ASSIGN_TO_KEYCACHE:
case SQLCOM_PRELOAD_KEYS:
case SQLCOM_GRANT:
case SQLCOM_GRANT_ROLE:
case SQLCOM_REVOKE:
case SQLCOM_REVOKE_ALL:
case SQLCOM_REVOKE_ROLE:
case SQLCOM_KILL:
case SQLCOM_ALTER_INSTANCE:
case SQLCOM_SET_ROLE:
case SQLCOM_ALTER_USER_DEFAULT_ROLE:
break;
case SQLCOM_CREATE_TABLE:
/*
CREATE TABLE ... START TRANSACTION is not supported with
prepared statements
*/
if (m_lex->create_info->m_transactional_ddl) {
my_error(ER_UNSUPPORTED_PS, MYF(0));
return true;
}
#if defined(__has_cpp_attribute)
#if __has_cpp_attribute(fallthrough)
[[fallthrough]];
#endif
#endif
case SQLCOM_SELECT:
case SQLCOM_DO:
case SQLCOM_DELETE:
case SQLCOM_DELETE_MULTI:
case SQLCOM_UPDATE:
case SQLCOM_UPDATE_MULTI:
case SQLCOM_INSERT:
case SQLCOM_INSERT_SELECT:
case SQLCOM_REPLACE:
case SQLCOM_REPLACE_SELECT:
case SQLCOM_CALL:
case SQLCOM_SHOW_BINLOG_EVENTS:
case SQLCOM_SHOW_BINLOGS:
case SQLCOM_SHOW_CHARSETS:
case SQLCOM_SHOW_COLLATIONS:
case SQLCOM_SHOW_CREATE_DB:
case SQLCOM_SHOW_CREATE_EVENT:
case SQLCOM_SHOW_CREATE_FUNC:
case SQLCOM_SHOW_CREATE_PROC:
case SQLCOM_SHOW_CREATE:
case SQLCOM_SHOW_CREATE_TRIGGER:
case SQLCOM_SHOW_CREATE_USER:
case SQLCOM_SHOW_DATABASES:
case SQLCOM_SHOW_ENGINE_LOGS:
case SQLCOM_SHOW_ENGINE_MUTEX:
case SQLCOM_SHOW_ENGINE_STATUS:
case SQLCOM_SHOW_ERRORS:
case SQLCOM_SHOW_EVENTS:
case SQLCOM_SHOW_FIELDS:
case SQLCOM_SHOW_FUNC_CODE:
case SQLCOM_SHOW_GRANTS:
case SQLCOM_SHOW_KEYS:
case SQLCOM_SHOW_BINLOG_STATUS:
case SQLCOM_SHOW_OPEN_TABLES:
case SQLCOM_SHOW_PLUGINS:
case SQLCOM_SHOW_PRIVILEGES:
case SQLCOM_SHOW_PROC_CODE:
case SQLCOM_SHOW_PROCESSLIST:
case SQLCOM_SHOW_PROFILE:
case SQLCOM_SHOW_PROFILES:
case SQLCOM_SHOW_RELAYLOG_EVENTS:
case SQLCOM_SHOW_REPLICAS:
case SQLCOM_SHOW_REPLICA_STATUS:
case SQLCOM_SHOW_STATUS:
case SQLCOM_SHOW_STATUS_PROC:
case SQLCOM_SHOW_STATUS_FUNC:
case SQLCOM_SHOW_STORAGE_ENGINES:
case SQLCOM_SHOW_TABLE_STATUS:
case SQLCOM_SHOW_TABLES:
case SQLCOM_SHOW_TRIGGERS:
case SQLCOM_SHOW_VARIABLES:
case SQLCOM_SET_RESOURCE_GROUP:
case SQLCOM_SHOW_WARNS:
res = m_lex->m_sql_cmd->prepare(thd);
break;
case SQLCOM_PREPARE:
case SQLCOM_EXECUTE:
case SQLCOM_DEALLOCATE_PREPARE:
default:
/*
Trivial check of all status commands and diagnostic commands.
This is easier than having things in the above case list,
as it's less chance for mistakes.
*/
if (!(sql_command_flags[sql_command] & CF_STATUS_COMMAND) ||
(sql_command_flags[sql_command] & CF_DIAGNOSTIC_STMT)) {
/* All other statements are not supported yet. */
my_error(ER_UNSUPPORTED_PS, MYF(0));
return true;
}
break;
}
if (res) return true;
trace_parameter_types(thd);
if (is_sql_prepare()) return false;
mem_root_deque<Item *> *types = nullptr;
Query_result *result = nullptr;
uint no_columns = 0;
if ((sql_command_flags[sql_command] & CF_HAS_RESULT_SET) &&
!m_lex->is_explain()) {
Query_expression *unit = m_lex->unit;
result = unit->query_result();
if (result == nullptr) result = unit->first_query_block()->query_result();
if (result == nullptr) result = m_lex->result;
types = unit->get_unit_column_types();
no_columns = result->field_count(*types);
}
return send_statement(thd, this, no_columns, result, types);
}
const char *fieldtype2str(enum enum_field_types type);
void Prepared_statement::trace_parameter_types(THD *thd) {
if (m_param_count == 0) return;
const Opt_trace_object anon(&thd->opt_trace);
Opt_trace_array typ(&thd->opt_trace, "statement_parameters");
Item_param **end = m_param_array + m_param_count;
char buf[50];
for (Item_param **it = m_param_array; it < end; ++it) {
enum_field_types t = (*it)->data_type();
const char *n = fieldtype2str(t);
switch (t) {
case MYSQL_TYPE_NEWDECIMAL:
case MYSQL_TYPE_TIME:
case MYSQL_TYPE_DATETIME:
snprintf(buf, sizeof(buf), "%s decimals=%d", n, (*it)->decimals);
n = buf;
default:;
}
typ.add_alnum(n);
}
}
/**
Initialize array of parameters in statement from LEX.
(We need to have quick access to items by number in mysql_stmt_get_longdata).
This is to avoid using malloc/realloc in the parser.
*/
static bool init_param_array(THD *thd, Prepared_statement *stmt) {
LEX *lex = stmt->m_lex;
if ((stmt->m_param_count = lex->param_list.elements)) {
if (stmt->m_param_count > static_cast<uint>(UINT_MAX16)) {
/* Error code to be defined in 5.0 */
my_error(ER_PS_MANY_PARAM, MYF(0));
return true;
}
Item_param **to;
List_iterator<Item_param> param_iterator(lex->param_list);
/* Use thd->mem_root as it points at statement mem_root */
stmt->m_param_array =
thd->mem_root->ArrayAlloc<Item_param *>(stmt->m_param_count);
if (stmt->m_param_array == nullptr) return true;
for (to = stmt->m_param_array;
to < stmt->m_param_array + stmt->m_param_count; ++to) {
*to = param_iterator++;
}
}
return false;
}
/**
COM_STMT_PREPARE handler.
Given a query string with parameter markers, create a prepared
statement from it and send PS info back to the client.
If parameter markers are found in the query, then store the information
using Item_param along with maintaining a list in lex->param_array, so
that a fast and direct retrieval can be made without going through all
field items.
In case of success a new statement id and metadata is sent
to the client, otherwise an error message is set in THD.
@param thd thread handle
@param query query to be prepared
@param length query string length, including ignored
trailing NULL or quote char.
@param stmt Prepared_statement to be used for preparation.
@note
This function parses the query and sends the total number of parameters
and resultset metadata information back to client (if any), without
executing the query i.e. without any log/disk writes. This allows the
queries to be re-executed without re-parsing during execute.
*/
void mysqld_stmt_prepare(THD *thd, const char *query, uint length,
Prepared_statement *stmt) {
DBUG_TRACE;
DBUG_PRINT("prep_query", ("%s", query));
assert(stmt != nullptr);
const bool switch_protocol = thd->is_classic_protocol();
if (switch_protocol) {
// set the current client capabilities before switching the protocol
thd->protocol_binary->set_client_capabilities(
thd->get_protocol()->get_client_capabilities());
thd->push_protocol(thd->protocol_binary);
}
// Initially, optimize the statement for the primary storage engine.
// If an eligible secondary storage engine is found, the statement
// may be reprepared for the secondary storage engine later.
thd->set_secondary_engine_optimization(
Secondary_engine_optimization::PRIMARY_TENTATIVELY);
/* Create PS table entry, set query text after rewrite. */
stmt->m_prepared_stmt =
MYSQL_CREATE_PS(stmt, stmt->m_id, thd->m_statement_psi, stmt->name().str,
stmt->name().length, nullptr, 0);
if (stmt->prepare(thd, query, length, nullptr)) {
/* Delete this stmt stats from PS table. */
MYSQL_DESTROY_PS(stmt->m_prepared_stmt);
/* Statement map deletes statement on erase */
thd->stmt_map.erase(stmt);
}
if (switch_protocol) thd->pop_protocol();
sp_cache_enforce_limit(thd->sp_proc_cache, stored_program_cache_size);
sp_cache_enforce_limit(thd->sp_func_cache, stored_program_cache_size);
// Prepared_statement::prepare_query() sends metadata packet if success
}
/**
Searches for the statement with the specified id and validates it.
@param thd [in] thread handle
@param com_data [in] command data
@param cmd [in] command type to be executed
@param stmt [out] pointer to Prepared_statement to store it if found
*/
bool mysql_stmt_precheck(THD *thd, const COM_DATA *com_data,
enum enum_server_command cmd,
Prepared_statement **stmt) {
*stmt = nullptr;
ulong stmt_id = 0;
switch (cmd) {
case COM_STMT_FETCH:
stmt_id = com_data->com_stmt_fetch.stmt_id;
if (!(*stmt = thd->stmt_map.find(stmt_id))) goto not_found;
break;
case COM_STMT_CLOSE:
stmt_id = com_data->com_stmt_close.stmt_id;
if (!(*stmt = thd->stmt_map.find(stmt_id))) goto silent_error;
break;
case COM_STMT_RESET: {
stmt_id = com_data->com_stmt_reset.stmt_id;
if (!(*stmt = thd->stmt_map.find(stmt_id))) goto not_found;
break;
}
case COM_STMT_PREPARE: {
if (!(*stmt = new Prepared_statement(thd)))
// out of memory: error is set in MEM_ROOT
goto silent_error; /* purecov: inspected */
if (thd->stmt_map.insert(*stmt))
/*
The error is set in the insert. The statement itself
will be also deleted there (this is how the hash works).
*/
goto silent_error;
break;
}
case COM_STMT_SEND_LONG_DATA: {
stmt_id = com_data->com_stmt_send_long_data.stmt_id;
if (!(*stmt = thd->stmt_map.find(stmt_id))) goto silent_error;
if (com_data->com_stmt_send_long_data.param_number >=
(*stmt)->m_param_count) {
/* Error will be sent in execute call */
(*stmt)->m_arena.set_state(Query_arena::STMT_ERROR);
(*stmt)->m_last_errno = ER_WRONG_ARGUMENTS;
sprintf((*stmt)->m_last_error, ER_THD(thd, ER_WRONG_ARGUMENTS),
"mysql_stmt_precheck");
goto silent_error;
}
break;
}
case COM_STMT_EXECUTE: {
stmt_id = com_data->com_stmt_execute.stmt_id;
if (!(*stmt = thd->stmt_map.find(stmt_id))) goto not_found;
if (thd->get_protocol()->has_client_capability(CLIENT_QUERY_ATTRIBUTES)) {
/*
For client supporting query attributes it's perfectly fine to send
more parameter values than the prepared statement has. The prepared
statement will take the first param_count values and will leave the
rest for the component service to consume. Thus altering the validity
check accordingly to just error out if there's less than the expected
number of parameters and pass if there's more.
*/
if ((*stmt)->m_param_count > com_data->com_stmt_execute.parameter_count)
goto wrong_arg;
} else {
if ((*stmt)->m_param_count !=
com_data->com_stmt_execute.parameter_count)
goto wrong_arg;
}
break;
}
default:
assert(0);
return true;
}
return false;
not_found:
char llbuf[22];
my_error(ER_UNKNOWN_STMT_HANDLER, MYF(0), static_cast<int>(sizeof(llbuf)),
llstr(stmt_id, llbuf), "mysql_stmt_precheck");
return true;
wrong_arg:
my_error(ER_WRONG_ARGUMENTS, MYF(0), "COM_STMT_EXECUTE");
return true;
silent_error:
return true;
}
/**
Get an SQL statement text from a user variable or from plain text.
If the statement is plain text, just assign the
pointers, otherwise allocate memory in thd->mem_root and copy
the contents of the variable, possibly with character
set conversion.
@param[in] lex main lex
@param[out] query_len length of the SQL statement (is set only
in case of success)
@retval
non-zero success
@retval
0 in case of error (out of memory)
*/
static const char *get_dynamic_sql_string(LEX *lex, size_t *query_len) {
THD *thd = lex->thd;
char *query_str = nullptr;
if (lex->prepared_stmt_code_is_varref) {
/* This is PREPARE stmt FROM or EXECUTE IMMEDIATE @var. */
String str;
const CHARSET_INFO *to_cs = thd->variables.collation_connection;
bool needs_conversion;
String *var_value = &str;
size_t unused;
size_t len;
/* Protects thd->user_vars */
mysql_mutex_lock(&thd->LOCK_thd_data);
const auto it = thd->user_vars.find(to_string(lex->prepared_stmt_code));
/*
Convert @var contents to string in connection character set. Although
it is known that int/real/NULL value cannot be a valid query we still
convert it for error messages to be uniform.
*/
if (it != thd->user_vars.end() && it->second->ptr()) {
user_var_entry *entry = it->second.get();
bool is_var_null;
var_value = entry->val_str(&is_var_null, &str, DECIMAL_NOT_SPECIFIED);
mysql_mutex_unlock(&thd->LOCK_thd_data);
/*
NULL value of variable checked early as entry->value so here
we can't get NULL in normal conditions
*/
assert(!is_var_null);
if (!var_value) goto end;
} else {
mysql_mutex_unlock(&thd->LOCK_thd_data);
/*
variable absent or equal to NULL, so we need to set variable to
something reasonable to get a readable error message during parsing
*/
str.set(STRING_WITH_LEN("NULL"), &my_charset_latin1);
}
needs_conversion = String::needs_conversion(
var_value->length(), var_value->charset(), to_cs, &unused);
len = (needs_conversion ? var_value->length() * to_cs->mbmaxlen
: var_value->length());
if (!(query_str = (char *)thd->mem_root->Alloc(len + 1))) goto end;
if (needs_conversion) {
uint dummy_errors;
len = copy_and_convert(query_str, len, to_cs, var_value->ptr(),
var_value->length(), var_value->charset(),
&dummy_errors);
} else
memcpy(query_str, var_value->ptr(), var_value->length());
query_str[len] = '\0'; // Safety (mostly for debug)
*query_len = len;
} else {
query_str = lex->prepared_stmt_code.str;
*query_len = lex->prepared_stmt_code.length;
}
end:
return query_str;
}
/**
SQLCOM_PREPARE implementation.
Prepare an SQL prepared statement. This is called from
mysql_execute_command and should therefore behave like an
ordinary query (e.g. should not reset any global THD data).
In case of success, OK packet is sent to the client,
otherwise an error message is set in THD.
@param thd thread handle
*/
void mysql_sql_stmt_prepare(THD *thd) {
LEX *lex = thd->lex;
const LEX_CSTRING &name = lex->prepared_stmt_name;
DBUG_TRACE;
Prepared_statement *stmt = thd->stmt_map.find_by_name(name);
if (stmt != nullptr) {
/*
If there is a statement with the same name, remove it. It is ok to
remove old and fail to insert a new one at the same time.
*/
if (stmt->is_in_use()) {
my_error(ER_PS_NO_RECURSION, MYF(0));
return;
}
MYSQL_DESTROY_PS(stmt->m_prepared_stmt);
stmt->deallocate(thd);
}
size_t query_len = 0;
const char *query = get_dynamic_sql_string(lex, &query_len);
if (query == nullptr) return; /* out of memory */
stmt = new Prepared_statement(thd);
if (stmt == nullptr) return; /* out of memory */
stmt->set_sql_prepare();
/* Set the name first, insert should know that this statement has a name */
if (stmt->set_name(name)) {
delete stmt;
return;
}
if (thd->stmt_map.insert(stmt)) {
/* The statement is deleted and an error is set if insert fails */
return;
}
/* Create PS table entry, set query text after rewrite. */
stmt->m_prepared_stmt =
MYSQL_CREATE_PS(stmt, stmt->id(), thd->m_statement_psi, stmt->name().str,
stmt->name().length, nullptr, 0);
if (stmt->prepare(thd, query, query_len, nullptr)) {
/* Delete this stmt stats from PS table. */
MYSQL_DESTROY_PS(stmt->m_prepared_stmt);
/* Statement map deletes the statement on erase */
thd->stmt_map.erase(stmt);
} else {
/* send the boolean tracker in the OK packet when
@@session_track_state_change is set to ON */
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, {});
my_ok(thd, 0L, 0L, "Statement prepared");
}
}
/**
Clears parameters from data left from previous execution or long data.
@param stmt prepared statement for which parameters should
be reset
*/
void reset_stmt_parameters(Prepared_statement *stmt) {
Item_param **item = stmt->m_param_array;
Item_param **end = item + stmt->m_param_count;
for (; item < end; ++item) {
(**item).reset();
(**item).sync_clones();
}
}
/**
COM_STMT_EXECUTE handler: execute a previously prepared statement.
If there are any parameters, then replace parameter markers with the
data supplied from the client, and then execute the statement.
This function uses binary protocol to send a possible result set
to the client.
In case of success OK packet or a result set is sent to the
client, otherwise an error message is set in THD.
@param thd current thread
@param stmt prepared statement
@param has_new_types true if parsed parameters have data types defined,
otherwise types from last execution will be used
@param execute_flags flag used to decide if a cursor should be used
@param parameters prepared statement's parsed parameters
*/
void mysqld_stmt_execute(THD *thd, Prepared_statement *stmt, bool has_new_types,
ulong execute_flags, PS_PARAM *parameters) {
DBUG_TRACE;
statement_id_to_session(thd);
#if defined(ENABLED_PROFILING)
thd->profiling->set_query_source(stmt->m_query_string.str,
stmt->m_query_string.length);
#endif
DBUG_PRINT("info", ("stmt: %p", stmt));
const bool switch_protocol = thd->is_classic_protocol();
if (switch_protocol) {
// set the current client capabilities before switching the protocol
thd->protocol_binary->set_client_capabilities(
thd->get_protocol()->get_client_capabilities());
thd->push_protocol(thd->protocol_binary);
}
MYSQL_EXECUTE_PS(thd->m_statement_psi, stmt->m_prepared_stmt);
// Initially, optimize the statement for the primary storage engine.
// If an eligible secondary storage engine is found, the statement
// may be reprepared for the secondary storage engine later.
thd->set_secondary_engine_optimization(
Secondary_engine_optimization::PRIMARY_TENTATIVELY);
MYSQL_SET_PS_SECONDARY_ENGINE(stmt->m_prepared_stmt, false);
// Query text for binary, general or slow log, if any of them is open
String expanded_query;
expanded_query.set_charset(default_charset_info);
// If no error happened while setting the parameters, execute statement.
if (!stmt->set_parameters(thd, &expanded_query, has_new_types, parameters)) {
const bool open_cursor = execute_flags & (ulong)CURSOR_TYPE_READ_ONLY;
stmt->execute_loop(thd, &expanded_query, open_cursor);
}
if (switch_protocol) thd->pop_protocol();
sp_cache_enforce_limit(thd->sp_proc_cache, stored_program_cache_size);
sp_cache_enforce_limit(thd->sp_func_cache, stored_program_cache_size);
/* Close connection socket; for use with client testing (Bug#43560). */
DBUG_EXECUTE_IF("close_conn_after_stmt_execute",
thd->get_protocol()->shutdown(););
}
/**
SQLCOM_EXECUTE implementation.
Execute prepared statement using parameter values from
lex->prepared_stmt_params and send result to the client using
text protocol. This is called from mysql_execute_command and
therefore should behave like an ordinary query (e.g. not change
global THD data, such as warning count, server status, etc).
This function uses text protocol to send a possible result set.
In case of success, OK (or result set) packet is sent to the
client, otherwise an error is set in THD.
@param thd thread handle
*/
void mysql_sql_stmt_execute(THD *thd) {
LEX *lex = thd->lex;
const LEX_CSTRING &name = lex->prepared_stmt_name;
DBUG_TRACE;
DBUG_PRINT("info", ("EXECUTE: %.*s\n", (int)name.length, name.str));
Prepared_statement *stmt = thd->stmt_map.find_by_name(name);
if (stmt == nullptr) {
my_error(ER_UNKNOWN_STMT_HANDLER, MYF(0), static_cast<int>(name.length),
name.str, "EXECUTE");
return;
}
if (stmt->m_param_count != lex->prepared_stmt_params.elements) {
my_error(ER_WRONG_ARGUMENTS, MYF(0), "EXECUTE");
return;
}
DBUG_PRINT("info", ("stmt: %p", stmt));
MYSQL_EXECUTE_PS(thd->m_statement_psi, stmt->m_prepared_stmt);
// Query text for binary, general or slow log, if any of them is open
String expanded_query;
if (stmt->set_parameters(thd, &expanded_query)) return;
stmt->execute_loop(thd, &expanded_query, false);
}
/**
COM_STMT_FETCH handler: fetches requested amount of rows from cursor.
@param thd Thread handle.
@param stmt Pointer to the prepared statement.
@param num_rows Number of rows to fetch.
*/
void mysqld_stmt_fetch(THD *thd, Prepared_statement *stmt, ulong num_rows) {
DBUG_TRACE;
thd->status_var.com_stmt_fetch++;
global_aggregated_stats.get_shard(thd->thread_id()).com_stmt_fetch++;
Server_side_cursor *cursor = stmt->m_cursor;
if (cursor == nullptr || !cursor->is_open()) {
my_error(ER_STMT_HAS_NO_OPEN_CURSOR, MYF(0), stmt->m_id);
return;
}
thd->stmt_arena = &stmt->m_arena;
Statement_backup stmt_backup;
stmt_backup.set_thd_to_ps(thd, stmt);
cursor->fetch(num_rows);
if (!cursor->is_open()) reset_stmt_parameters(stmt);
stmt_backup.restore_thd(thd, stmt);
thd->stmt_arena = thd;
}
/**
Reset a prepared statement in case there was a recoverable error.
This function resets statement to the state it was right after prepare.
It can be used to:
- clear an error happened during mysqld_stmt_send_long_data
- cancel long data stream for all placeholders without
having to call mysqld_stmt_execute.
- close an open cursor
Sends 'OK' packet in case of success (statement was reset)
or 'ERROR' packet (unrecoverable error/statement not found/etc).
@param thd Thread handle
@param stmt Pointer to the Prepared_statement
*/
void mysqld_stmt_reset(THD *thd, Prepared_statement *stmt) {
DBUG_TRACE;
thd->status_var.com_stmt_reset++;
global_aggregated_stats.get_shard(thd->thread_id()).com_stmt_reset++;
stmt->close_cursor();
/*
Clear parameters from data which could be set by
mysqld_stmt_send_long_data() call.
*/
reset_stmt_parameters(stmt);
stmt->m_arena.set_state(Query_arena::STMT_PREPARED);
query_logger.general_log_print(thd, thd->get_command(), NullS);
my_ok(thd);
}
/**
Delete a prepared statement from memory.
@note
we don't send any reply to this command.
*/
void mysqld_stmt_close(THD *thd, Prepared_statement *stmt) {
DBUG_TRACE;
/*
The only way currently a statement can be deallocated when it's
in use is from within Dynamic SQL.
*/
assert(!stmt->is_in_use());
MYSQL_DESTROY_PS(stmt->m_prepared_stmt);
stmt->deallocate(thd);
query_logger.general_log_print(thd, thd->get_command(), NullS);
}
/**
SQLCOM_DEALLOCATE implementation.
Close an SQL prepared statement. As this can be called from Dynamic
SQL, we should be careful to not close a statement that is currently
being executed.
OK packet is sent in case of success, otherwise an error
message is set in THD.
*/
void mysql_sql_stmt_close(THD *thd) {
const LEX_CSTRING &name = thd->lex->prepared_stmt_name;
DBUG_PRINT("info",
("DEALLOCATE PREPARE: %.*s\n", (int)name.length, name.str));
Prepared_statement *stmt = thd->stmt_map.find_by_name(name);
if (stmt == nullptr) {
my_error(ER_UNKNOWN_STMT_HANDLER, MYF(0), static_cast<int>(name.length),
name.str, "DEALLOCATE PREPARE");
return;
}
if (stmt->is_in_use()) {
my_error(ER_PS_NO_RECURSION, MYF(0));
return;
}
MYSQL_DESTROY_PS(stmt->m_prepared_stmt);
stmt->deallocate(thd);
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, {});
my_ok(thd);
}
/**
Handle long data in pieces from client.
Get a part of a long data. To make the protocol efficient, we are
not sending any return packets here. If something goes wrong, then
we will send the error on 'execute' We assume that the client takes
care of checking that all parts are sent to the server. (No checking
that we get a 'end of column' in the server is performed).
@param thd Thread handle
@param stmt Pointer to Prepared_statement
@param param_number Number of parameters
@param str String to append
@param length Length of string (including end \\0)
*/
void mysql_stmt_get_longdata(THD *thd, Prepared_statement *stmt,
uint param_number, uchar *str, ulong length) {
DBUG_TRACE;
thd->status_var.com_stmt_send_long_data++;
global_aggregated_stats.get_shard(thd->thread_id()).com_stmt_send_long_data++;
Diagnostics_area new_stmt_da(false);
thd->push_diagnostics_area(&new_stmt_da);
Item_param *param = stmt->m_param_array[param_number];
param->set_longdata((char *)str, length);
if (thd->get_stmt_da()->is_error()) {
stmt->m_arena.set_state(Query_arena::STMT_ERROR);
stmt->m_last_errno = thd->get_stmt_da()->mysql_errno();
snprintf(stmt->m_last_error, sizeof(stmt->m_last_error), "%.*s",
MYSQL_ERRMSG_SIZE - 1, thd->get_stmt_da()->message_text());
}
thd->pop_diagnostics_area();
query_logger.general_log_print(thd, thd->get_command(), NullS);
}
/***************************************************************************
Query_fetch_protocol_binary
****************************************************************************/
namespace {
bool Query_fetch_protocol_binary::send_result_set_metadata(
THD *thd, const mem_root_deque<Item *> &list, uint flags) {
bool rc;
protocol.set_client_capabilities(
thd->get_protocol()->get_client_capabilities());
/*
Protocol::send_result_set_metadata caches the information about column
types: this information is later used to send data. Therefore, the same
dedicated Protocol object must be used for all operations with
a cursor.
*/
thd->push_protocol(&protocol);
rc = Query_result_send::send_result_set_metadata(thd, list, flags);
thd->pop_protocol();
return rc;
}
bool Query_fetch_protocol_binary::send_eof(THD *thd) {
/*
Don't send EOF if we're in error condition (which implies we've already
sent or are sending an error)
*/
if (thd->is_error()) return true;
::my_eof(thd);
return false;
}
bool Query_fetch_protocol_binary::send_data(
THD *thd, const mem_root_deque<Item *> &fields) {
bool rc;
// set the current client capabilities before switching the protocol
protocol.set_client_capabilities(
thd->get_protocol()->get_client_capabilities());
thd->push_protocol(&protocol);
rc = Query_result_send::send_data(thd, fields);
thd->pop_protocol();
return rc;
}
} // namespace
/*******************************************************************
* Reprepare_observer
*******************************************************************/
/** Push an error to the error stack and return true for now. */
bool Reprepare_observer::report_error(THD *thd) {
/*
This 'error' is purely internal to the server:
- No exception handler is invoked,
- No condition is added in the condition area (warn_list).
The Diagnostics Area is set to an error status to enforce
that this thread execution stops and returns to the caller,
backtracking all the way to Prepared_statement::execute_loop().
As the DA has not yet been reset at this point, we'll need to
reset the previous statement's result status first.
Test with rpl_sp_effects and friends.
*/
thd->get_stmt_da()->reset_diagnostics_area();
thd->get_stmt_da()->set_error_status(thd, ER_NEED_REPREPARE);
m_invalidated = true;
m_attempt++;
return true;
}
/**
Requests for repreparation of statement.
@returns true if request has been placed.
*/
bool ask_to_reprepare(THD *thd) {
Reprepare_observer *reprepare_observer = thd->get_reprepare_observer();
if (reprepare_observer != nullptr && reprepare_observer->report_error(thd)) {
assert(thd->is_error());
return true;
}
return false;
}
/***************************************************************************
Prepared_statement
****************************************************************************/
Prepared_statement::Prepared_statement(THD *thd_arg)
: m_arena(&m_mem_root, Query_arena::STMT_INITIALIZED),
m_id(++thd_arg->statement_id_counter),
m_mem_root(key_memory_prepared_statement_main_mem_root,
thd_arg->variables.query_alloc_block_size) {
*m_last_error = '\0';
}
void Prepared_statement::close_cursor() {
if (m_cursor == nullptr) return;
m_cursor->close();
m_used_as_cursor = false;
}
void Prepared_statement::setup_stmt_logging(THD *thd) {
DBUG_EXECUTE_IF("bug16617026_simulate_audit_log_ps",
{ m_lex->safe_to_cache_query = 0; });
/*
Decide if we have to expand the statement (because we must write it to logs)
or not.
We don't have to substitute the params when bin-logging DML in RBL.
*/
if ((mysql_bin_log.is_open() && is_update_query(m_lex->sql_command) &&
(!thd->is_current_stmt_binlog_format_row() ||
((sql_command_flags[m_lex->sql_command] & CF_AUTO_COMMIT_TRANS) ==
CF_AUTO_COMMIT_TRANS))) ||
opt_general_log || opt_slow_log ||
(m_lex->sql_command == SQLCOM_SELECT && m_lex->safe_to_cache_query &&
!m_lex->is_explain()) ||
is_global_audit_mask_set()) {
m_with_log = true;
}
/*
@todo WL#6570
In the if() above there is is_global_audit_mask_set()) .
Because caching_sha2 is loaded by default, it this function return true
which makes with_log always true.
If not desirable, we can limit the test to only these plugin classes:
MYSQL_AUDIT_GENERAL_CLASS, MYSQL_AUDIT_PARSE_CLASS,
MYSQL_AUDIT_TABLE_ACCESS_CLASS, MYSQL_AUDIT_QUERY_CLASS,
MYSQL_AUDIT_STORED_PROGRAM_CLASS, MYSQL_AUDIT_AUTHENTICATION_CLASS.
E.g. the plugin which mandated this change (bug#16617026) was about
Enterprise audit log plugin which is of class MYSQL_AUDIT_GENERAL_CLASS.
Marek said.
Another related issue: in SQL PREPARE/EXECUTE
Item_param::set_type_actual() is only called in the "if (m_with_log)" branch
of insert_parameters_from_vars(); so, if m_with_log is false, it is
skipped. Testcase:
- comment-out the condition is_global_audit_mask_set() above, recompile
- start with --disable-log-in, no general log, no slow log
- with that and RAND() below (=>safe_to_cache_query=false),
m_with_log is false.
- prepare s from "select ?, rand()";
set @a:="ab";
execute s using @a; execution fails.
*/
}
/**
Destroy this prepared statement, cleaning up all used memory
and resources.
This is called from @c deallocate() to handle COM_STMT_CLOSE and
DEALLOCATE PREPARE or when THD ends and all prepared statements are freed.
*/
Prepared_statement::~Prepared_statement() {
DBUG_TRACE;
DBUG_PRINT("enter", ("stmt: %p cursor: %p", this, m_cursor));
if (m_used_as_cursor) {
close_cursor();
}
if (m_regular_result != nullptr) ::destroy_at(m_regular_result);
if (m_cursor_result != nullptr) ::destroy_at(m_cursor_result);
if (m_aux_result != nullptr) ::destroy_at(m_aux_result);
m_cursor = nullptr;
m_cursor_result = nullptr;
m_regular_result = nullptr;
m_aux_result = nullptr;
/*
We have to call free on the items even if cleanup is called as some items,
like Item_param, don't free everything until free_items()
*/
m_arena.free_items();
if (m_lex != nullptr) {
assert(m_lex->sphead == nullptr);
lex_end(m_lex);
m_lex->destroy();
delete pointer_cast<st_lex_local *>(m_lex); // TRASH memory
}
}
void Prepared_statement::cleanup_stmt(THD *thd) {
DBUG_TRACE;
DBUG_PRINT("enter", ("stmt: %p", this));
cleanup_items(m_arena.item_list());
thd->cleanup_after_query();
}
bool Prepared_statement::set_name(const LEX_CSTRING &name_arg) {
m_name.length = name_arg.length;
m_name.str = static_cast<char *>(
memdup_root(m_arena.mem_root, name_arg.str, name_arg.length));
return m_name.str == nullptr;
}
/**
Remember the current database.
We must reset/restore the current database during execution of
a prepared statement since it affects execution environment:
privileges, @@character_set_database, and other.
@return Returns an error if out of memory.
*/
bool Prepared_statement::set_db(const LEX_CSTRING &db_arg) {
/* Remember the current database. */
if (db_arg.str && db_arg.length) {
m_db.str = m_arena.strmake(db_arg.str, db_arg.length);
m_db.length = db_arg.length;
} else {
m_db = NULL_CSTR;
}
return db_arg.str != nullptr && m_db.str == nullptr;
}
/**************************************************************************
Common parts of mysql_[sql]_stmt_prepare, mysql_[sql]_stmt_execute.
Essentially, these functions do all the magic of preparing/executing
a statement, leaving network communication, input data handling and
global THD state management to the caller.
***************************************************************************/
/**
Parse statement text, validate the statement, and prepare it for execution.
You should not change global THD state in this function, if at all
possible: it may be called from any context, e.g. when executing
a COM_* command, and SQLCOM_* command, or a stored procedure.
@param thd thread handle
@param query_str Statement text
@param query_length Length of statement string
@param orig_param_array Array containing pointers to parameter items
that contain data type information for query.
This is used during reprepare.
= NULL: Derive parameter metadata from query only.
@note
Precondition:
The caller must ensure that thd->change_list and thd->item_list
is empty: this function will not back them up but will free
in the end of its execution.
@note
Postcondition:
thd->mem_root contains unused memory allocated during validation.
*/
bool Prepared_statement::prepare(THD *thd, const char *query_str,
size_t query_length,
Item_param **orig_param_array) {
bool error;
Query_arena arena_backup;
Query_arena *old_stmt_arena;
sql_digest_state *parent_digest = thd->m_digest;
PSI_statement_locker *parent_locker = thd->m_statement_psi;
unsigned char *token_array = nullptr;
DBUG_TRACE;
/*
If this is an SQLCOM_PREPARE, we also increase Com_prepare_sql.
However, it seems handy if com_stmt_prepare is increased always,
no matter what kind of prepare is processed.
*/
thd->status_var.com_stmt_prepare++;
global_aggregated_stats.get_shard(thd->thread_id()).com_stmt_prepare++;
assert(m_lex == nullptr);
m_lex = new (m_arena.mem_root) st_lex_local;
if (m_lex == nullptr) return true;
if (set_db(thd->db())) return true;
/*
alloc_query() uses thd->memroot && thd->query, so we should call
both of backup_statement() and backup_query_arena() here.
*/
Statement_backup stmt_backup;
stmt_backup.set_thd_to_ps(thd, this);
stmt_backup.save_rlb(thd);
thd->swap_query_arena(m_arena, &arena_backup);
if (alloc_query(thd, query_str, query_length)) {
stmt_backup.restore_thd(thd, this);
stmt_backup.restore_rlb(thd);
thd->swap_query_arena(arena_backup, &m_arena);
return true;
}
if (max_digest_length > 0) {
token_array = (unsigned char *)thd->alloc(max_digest_length);
}
old_stmt_arena = thd->stmt_arena;
thd->stmt_arena = &m_arena;
Parser_state parser_state;
if (parser_state.init(thd, thd->query().str, thd->query().length)) {
stmt_backup.restore_thd(thd, this);
stmt_backup.restore_rlb(thd);
thd->swap_query_arena(arena_backup, &m_arena);
thd->stmt_arena = old_stmt_arena;
return true;
}
parser_state.m_lip.stmt_prepare_mode = true;
parser_state.m_lip.multi_statements = false;
lex_start(thd);
m_lex->context_analysis_only |= CONTEXT_ANALYSIS_ONLY_PREPARE;
thd->m_digest = nullptr;
thd->m_statement_psi = nullptr;
sql_digest_state digest;
digest.reset(token_array, max_digest_length);
thd->m_digest = &digest;
parser_state.m_input.m_has_digest = true;
// we produce digest if it's not explicitly turned off
// by setting maximum digest length to zero
if (get_max_digest_length() != 0)
parser_state.m_input.m_compute_digest = true;
thd->m_parser_state = &parser_state;
invoke_pre_parse_rewrite_plugins(thd);
thd->m_parser_state = nullptr;
error = thd->is_error();
if (!error) {
error = parse_sql(thd, &parser_state, nullptr);
}
error |= thd->is_error();
if (!error) { // We've just created the statement maybe there is a rewrite
invoke_post_parse_rewrite_plugins(thd, true);
error = init_param_array(thd, this);
}
error |= thd->is_error();
// Bind Sql command object with this prepared statement
if (m_lex->m_sql_cmd != nullptr) m_lex->m_sql_cmd->set_owner(this);
m_lex->set_trg_event_type_for_tables();
/*
Pre-clear the diagnostics area unless a warning was thrown
during parsing.
*/
if (thd->lex->keep_diagnostics != DA_KEEP_PARSE_ERROR)
thd->get_stmt_da()->reset_condition_info(thd);
/*
While doing context analysis of the query (in prepare_query())
we allocate a lot of additional memory: for open tables, JOINs, derived
tables, etc. Let's save a snapshot of current parse tree to the
statement and restore original THD. In cases when some tree
transformation can be reused on execute, we set again thd->mem_root from
stmt->mem_root (see setup_wild for one place where we do that).
*/
thd->swap_query_arena(arena_backup, &m_arena);
/*
If called from a stored procedure, ensure that we won't rollback
external changes when cleaning up after validation.
*/
assert(thd->change_list.is_empty());
/*
Marker used to release metadata locks acquired while the prepared
statement is being checked.
*/
MDL_savepoint mdl_savepoint = thd->mdl_context.mdl_savepoint();
// A subsystem, such as the Audit plugin, may have set error unnoticed:
error |= thd->is_error();
if (orig_param_array != nullptr) copy_parameter_types(orig_param_array);
/*
The only case where we should have items in the thd->item_list is
after stmt->set_params_from_vars(), which may in some cases create
Item_null objects.
*/
if (error == 0) {
// Update system variables specified in SET_VAR hints while preparing
if (m_lex->opt_hints_global && m_lex->opt_hints_global->sys_var_hint)
m_lex->opt_hints_global->sys_var_hint->update_vars(thd);
error = prepare_query(thd);
if (m_lex->opt_hints_global && m_lex->opt_hints_global->sys_var_hint)
m_lex->opt_hints_global->sys_var_hint->restore_vars(thd);
}
assert(error || !thd->is_error());
/*
Currently CREATE PROCEDURE/TRIGGER/EVENT are prohibited in prepared
statements: ensure we have no memory leak here if by someone tries
to PREPARE stmt FROM "CREATE PROCEDURE ..."
*/
assert(m_lex->sphead == nullptr || error != 0);
/* The order is important */
m_lex->cleanup(true);
m_lex->clear_values_map();
close_thread_tables(thd);
thd->mdl_context.rollback_to_savepoint(mdl_savepoint);
/*
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.
Once dynamic SQL is allowed as substatements the below if-statement
has to be adjusted to not do rollback in substatement.
*/
assert(!thd->in_sub_stmt);
if (thd->transaction_rollback_request) {
trans_rollback_implicit(thd);
thd->mdl_context.release_transactional_locks();
}
lex_end(m_lex);
rewrite_query(thd);
const char *display_query_string;
int display_query_length;
if (thd->rewritten_query().length()) {
display_query_string = thd->rewritten_query().ptr();
display_query_length = thd->rewritten_query().length();
} else {
display_query_string = thd->query().str;
display_query_length = thd->query().length;
}
thd->set_query_for_display(display_query_string, display_query_length);
MYSQL_SET_PS_TEXT(m_prepared_stmt, display_query_string,
display_query_length);
cleanup_stmt(thd);
stmt_backup.restore_thd(thd, this);
thd->stmt_arena = old_stmt_arena;
if (error == 0) {
setup_stmt_logging(thd);
m_lex->context_analysis_only &= ~CONTEXT_ANALYSIS_ONLY_PREPARE;
m_arena.set_state(Query_arena::STMT_PREPARED);
m_in_use = false;
/*
Log COM_STMT_PREPARE to the general log. Note, that in case of SQL
prepared statements this causes two records to be output:
Query PREPARE stmt from @user_variable
Prepare <statement SQL text>
This is considered user-friendly, since in the second log Entry
we output the actual statement text rather than the variable name.
Rewriting/password obfuscation:
- If we're preparing from a string literal rather than from a
variable, the literal is elided in the "Query" log line, as
it may contain a password. (As we've parsed the PREPARE statement,
but not the statement to prepare yet, we don't know at that point.)
Eliding the literal is fine, as we'll print it in the next log line
("Prepare"), anyway.
- Any passwords in the "Prepare" line should be substituted with their
hashes, or a notice.
Do not print anything if this is an SQL prepared statement and
we're inside a stored procedure (also called Dynamic SQL) --
sub-statements inside stored procedures are not logged into
the general log.
*/
if (thd->sp_runtime_ctx == nullptr) {
if (thd->rewritten_query().length())
query_logger.general_log_write(thd, COM_STMT_PREPARE,
thd->rewritten_query().ptr(),
thd->rewritten_query().length());
else
query_logger.general_log_write(
thd, COM_STMT_PREPARE, m_query_string.str, m_query_string.length);
/* audit plugins can return an error */
error |= thd->is_error();
}
}
/* Restore the original rewritten query. */
stmt_backup.restore_rlb(thd);
thd->m_digest = parent_digest;
thd->m_statement_psi = parent_locker;
return error;
}
bool Prepared_statement::set_parameters(THD *thd, String *expanded_query,
bool has_new_types,
PS_PARAM *parameters) {
return set_parameters(thd, expanded_query, has_new_types, parameters,
enum_param_pack_type::PACKED);
}
bool Prepared_statement::set_parameters(THD *thd, String *expanded_query,
bool has_new_types,
PS_PARAM *parameters,
enum enum_param_pack_type pack_type) {
if (m_param_count == 0) return false;
if (insert_parameters(thd, expanded_query, has_new_types, parameters,
pack_type)) {
reset_stmt_parameters(this);
return true;
}
return false;
}
/**
Assign parameter values from specified variables..
@param thd current thread
@param expanded_query a container with the original SQL statement.
'?' placeholders will be replaced with
their values in case of success.
The result is used for logging and replication
@returns false if success, true if error
(likely a conversion error or out of memory)
*/
bool Prepared_statement::set_parameters(THD *thd, String *expanded_query) {
if (insert_parameters_from_vars(thd, thd->lex->prepared_stmt_params,
expanded_query)) {
reset_stmt_parameters(this);
return true;
}
return false;
}
/**
Disables the general log for the current session by setting the OPTION_LOG_OFF
bit in thd->variables.option_bits.
@param thd the session
@return whether the setting was changed
@retval false if the general log was already disabled for this session
@retval true if the general log was enabled for the session and is now
disabled
*/
static bool disable_general_log(THD *thd) {
if ((thd->variables.option_bits & OPTION_LOG_OFF) != 0) return false;
thd->variables.option_bits |= OPTION_LOG_OFF;
return true;
}
/**
Check resolved parameter types versus actual parameter types.
Assumes that parameter values have been assigned to the parameters.
@returns true if parameter types are compatible, false otherwise.
*/
bool Prepared_statement::check_parameter_types() {
Item_param **end = m_param_array + m_param_count;
for (Item_param **it = m_param_array; it < end; ++it) {
Item_param *const item = *it;
assert(item->param_state() != Item_param::NO_VALUE);
/*
- An inherited type is always accepted, it is like a dynamic cast and
a runtime check is needed.
- A pinned type is always accepted and not force a reprepare. However,
the actual type must be checked for consistency when parameters values
are analyzed.
- The NULL value is accepted for all parameter types.
*/
if (item->is_type_inherited() || item->is_type_pinned() ||
item->param_state() == Item_param::NULL_VALUE)
continue;
/*
It is expected that all string values may be cast to the desired data
type (but errors may be reported during this conversion).
So we do not reprepare in that case, with one exception. Consider:
col = param, where 'col' is of an signed integer type; and so 'param' is
BIGINT SIGNED. Assume that a string is passed for 'param', like
'18446744073709551615'. In a conventional, non-prepared execution,
comparison would be done as DOUBLE and would yield FALSE. In prepared
execution, conversion of this string to signed BIGINT means that 'param'
would be -1, and the comparison could yield TRUE. This is dealt with
below.
*/
if (item->data_type_actual() == MYSQL_TYPE_VARCHAR) {
if (item->result_type() == INT_RESULT) {
// First evaluate as DECIMAL
String *s = item->val_str(nullptr);
my_decimal decimal_value;
int result = str2my_decimal(0, s->ptr(), s->length(), s->charset(),
&decimal_value);
if (result == E_DEC_TRUNCATED || result == E_DEC_BAD_NUM) {
// Garbage in string, execution proceeds; at evaluation time val_int()
// will send a warning/error, which is not the present function's job.
continue;
}
// If result==E_DEC_OVERFLOW, decimal_value was rounded to a value
// (+-9999...) which is larger than any integer, so the comparison with
// 'col' will be correct. So we needn't handle this case
// specially.
if (result == E_DEC_OK) {
longlong i; // Convert to integer, just to see if it is in range
result = my_decimal2int(0, &decimal_value, item->unsigned_flag, &i);
}
if (result != E_DEC_OK) {
// Value is outside integer range; re-prepare as DECIMAL
item->set_decimal(&decimal_value);
return false;
}
}
continue;
}
switch (item->data_type()) {
case MYSQL_TYPE_BOOL:
case MYSQL_TYPE_TINY:
case MYSQL_TYPE_SHORT:
case MYSQL_TYPE_INT24:
case MYSQL_TYPE_LONG:
case MYSQL_TYPE_LONGLONG:
case MYSQL_TYPE_BIT:
case MYSQL_TYPE_YEAR:
/*
When resolved type is integer, accept only integer values.
Accept only values of same signedness, otherwise actual values may
be out of range.
Exception: YEAR values must be range-checked anyway, so they are
allowed, regardless of signedness.
*/
if (item->data_type_actual() != MYSQL_TYPE_LONGLONG ||
(item->data_type() != MYSQL_TYPE_YEAR &&
item->unsigned_flag != item->is_unsigned_actual()))
return false;
break;
case MYSQL_TYPE_NEWDECIMAL:
/*
Parameters of type DECIMAL have large precisions, so they can
also accommodate any integer values, both signed and unsigned.
*/
assert(item->decimal_precision() - item->decimals >= 20);
if (item->data_type_actual() != MYSQL_TYPE_LONGLONG &&
item->data_type_actual() != MYSQL_TYPE_NEWDECIMAL)
return false;
break;
case MYSQL_TYPE_FLOAT:
case MYSQL_TYPE_DOUBLE:
/*
Parameters of floating-point type accept also integer and
decimal values. Rounding errors may occur during conversion to
DOUBLE, but this is acceptable since it has already been inferred
that the function using the parameter treats its arguments as
floating-point.
*/
if (item->data_type_actual() != MYSQL_TYPE_LONGLONG &&
item->data_type_actual() != MYSQL_TYPE_NEWDECIMAL &&
item->data_type_actual() != MYSQL_TYPE_DOUBLE)
return false;
break;
case MYSQL_TYPE_DATE:
/*
Allow temporal values to be provided as numbers.
(a) date_expr = numeric *literal* has a special behaviour, the
numeric literal is converted to a date (e.g. 20010101 -> 2001-01-01)
and arguments are compared as DATE.
(b) date_expr = other numeric expression, doesn't have this (so
arguments get compared as DOUBLE).
(a) and (b) differ in this subtle case: if the number is 00010101 it
is 10101 and thus as date it is a two-year date so converted to
20010101. While as DOUBLE it is 10101. So in one case it matches
the date 2001-01-01 and in the other it matches 0001-01-01.
We want {date_expr = ? , pass a number for ?}
to behave like (a). Without the special case below, it would cause a
re-preparation and then be treated like (b).
*/
if (item->data_type_actual() == MYSQL_TYPE_LONGLONG ||
item->data_type_actual() == MYSQL_TYPE_NEWDECIMAL ||
item->data_type_actual() == MYSQL_TYPE_DOUBLE ||
item->data_type_actual() == MYSQL_TYPE_DATE)
continue;
// If actual type is TIME or DATETIME, force a reprepare as DATETIME
if (item->data_type_actual() == MYSQL_TYPE_DATETIME ||
item->data_type_actual() == MYSQL_TYPE_TIME)
return false;
break;
case MYSQL_TYPE_TIME:
if (item->data_type_actual() == MYSQL_TYPE_LONGLONG ||
item->data_type_actual() == MYSQL_TYPE_NEWDECIMAL ||
item->data_type_actual() == MYSQL_TYPE_DOUBLE ||
item->data_type_actual() == MYSQL_TYPE_TIME ||
item->data_type_actual() == MYSQL_TYPE_DATETIME)
continue;
/*
Actual type DATETIME: use the time component (sic)
Actual type DATE: reprepare as DATETIME
*/
if (item->data_type_actual() == MYSQL_TYPE_DATE) {
return false;
}
break;
case MYSQL_TYPE_DATETIME:
if (item->data_type_actual() == MYSQL_TYPE_LONGLONG ||
item->data_type_actual() == MYSQL_TYPE_NEWDECIMAL ||
item->data_type_actual() == MYSQL_TYPE_DOUBLE ||
item->data_type_actual() == MYSQL_TYPE_DATETIME ||
item->data_type_actual() == MYSQL_TYPE_DATE ||
item->data_type_actual() == MYSQL_TYPE_TIME)
continue;
break;
case MYSQL_TYPE_VARCHAR:
case MYSQL_TYPE_JSON:
case MYSQL_TYPE_TINY_BLOB:
case MYSQL_TYPE_MEDIUM_BLOB:
case MYSQL_TYPE_BLOB:
case MYSQL_TYPE_LONG_BLOB:
if (item->data_type_actual() == MYSQL_TYPE_LONGLONG ||
item->data_type_actual() == MYSQL_TYPE_NEWDECIMAL ||
item->data_type_actual() == MYSQL_TYPE_DOUBLE ||
item->data_type_actual() == MYSQL_TYPE_DATETIME ||
item->data_type_actual() == MYSQL_TYPE_TIMESTAMP ||
item->data_type_actual() == MYSQL_TYPE_DATE ||
item->data_type_actual() == MYSQL_TYPE_TIME)
return false;
break;
default:
assert(false);
}
}
return true;
}
/**
Execute a prepared statement. Re-prepare it a limited number
of times if necessary.
Try to execute a prepared statement. If there is a metadata
validation error, prepare a new copy of the prepared statement,
swap the old and the new statements, and try again.
If there is a validation error again, repeat the above, but
perform not more than a maximum number of times. Reprepare_observer
ensures that a prepared statement execution is retried not more than a
maximum number of times.
@note We have to try several times in a loop since we
release metadata locks on tables after prepared statement
prepare. Therefore, a DDL statement may sneak in between prepare
and execute of a new statement. If a prepared statement execution
is retried for a maximum number of times then we give up.
@param thd current thread.
@param expanded_query Query string.
@param open_cursor Flag to specify if a cursor should be used.
@return a bool value representing the function execution status.
@retval true error: either statement execution is retried for
a maximum number of times or some general error.
@retval false successfully executed the statement, perhaps
after having reprepared it a few times.
*/
bool Prepared_statement::execute_loop(THD *thd, String *expanded_query,
bool open_cursor) {
Reprepare_observer reprepare_observer;
bool error;
bool reprepared_for_types [[maybe_unused]] = false;
auto scope_guard = create_scope_guard(
[thd] { thd->set_secondary_engine_statement_context(nullptr); });
/* Check if we got an error when sending long data */
if (m_arena.get_state() == Query_arena::STMT_ERROR) {
my_message(m_last_errno, m_last_error, MYF(0));
return true;
}
assert(!thd->get_stmt_da()->is_set());
if (unlikely(!thd->security_context()->account_is_locked() &&
thd->security_context()->password_expired() &&
m_lex->sql_command != SQLCOM_SET_PASSWORD &&
m_lex->sql_command != SQLCOM_ALTER_USER)) {
my_error(ER_MUST_CHANGE_PASSWORD, MYF(0));
return true;
}
if (m_lex->m_sql_cmd != nullptr) {
m_lex->m_sql_cmd->enable_secondary_storage_engine();
}
// Remember if the general log was temporarily disabled when repreparing the
// statement for a secondary engine.
bool general_log_temporarily_disabled = false;
// Reprepare statement unconditionally if it contains UDF references
if (m_lex->has_udf() && reprepare(thd)) {
return true;
}
// Reprepare statement if protocol has changed.
// Note: this is not possible in current code base, hence the assert.
if (m_active_protocol != nullptr &&
m_active_protocol != thd->get_protocol()) {
assert(false);
if (reprepare(thd)) return true;
}
// Some SQL commands need re-preparation, such as Sql_cmd_create_table
// when the keys involve an expression.
if (!m_first_execution && m_lex->m_sql_cmd &&
m_lex->m_sql_cmd->reprepare_on_execute_required()) {
if (reprepare(thd)) return true;
}
reexecute:
/*
If the item_list is not empty, we'll wrongly free some externally
allocated items when cleaning up after validation of the prepared
statement.
*/
assert(thd->item_list() == nullptr);
if (!check_parameter_types()) {
// Only one reprepare is required in case of parameter mismatch
assert(!reprepared_for_types);
reprepared_for_types = true;
if (reprepare(thd)) return true;
goto reexecute;
}
reprepared_for_types = false;
/*
Install the metadata observer. If some metadata version is
different from prepare time and an observer is installed,
the observer method will be invoked to push an error into
the error stack.
*/
Reprepare_observer *stmt_reprepare_observer = nullptr;
if (sql_command_flags[m_lex->sql_command] & CF_REEXECUTION_FRAGILE) {
reprepare_observer.reset_reprepare_observer();
stmt_reprepare_observer = &reprepare_observer;
}
thd->push_reprepare_observer(stmt_reprepare_observer);
DEBUG_SYNC(thd, "before_statement_execute");
error = execute(thd, expanded_query, open_cursor) || thd->is_error();
thd->pop_reprepare_observer();
// Check if we have a non-fatal error and the statement allows reexecution.
if ((sql_command_flags[m_lex->sql_command] & CF_REEXECUTION_FRAGILE) &&
error && !thd->is_fatal_error() && !thd->is_killed()) {
// If we have an error due to a metadata change, reprepare the
// statement and execute it again.
if (reprepare_observer.is_invalidated()) {
assert(thd->get_stmt_da()->mysql_errno() == ER_NEED_REPREPARE);
if (reprepare_observer.can_retry()) {
thd->clear_error();
error = reprepare(thd);
DEBUG_SYNC(thd, "after_statement_reprepare");
} else {
/*
Reprepare_observer sets error status in DA but Sql_condition is not
added. Please check Reprepare_observer::report_error(). Pushing
Sql_condition for ER_NEED_REPREPARE here.
*/
Diagnostics_area *da = thd->get_stmt_da();
da->push_warning(thd, da->mysql_errno(), da->returned_sqlstate(),
Sql_condition::SL_ERROR, da->message_text());
}
} else {
// Otherwise, if repreparation was requested, try again in the primary
// or secondary engine, depending on cause.
const uint err_seen = thd->get_stmt_da()->mysql_errno();
if (err_seen == ER_PREPARE_FOR_PRIMARY_ENGINE ||
err_seen == ER_PREPARE_FOR_SECONDARY_ENGINE) {
assert(thd->secondary_engine_optimization() ==
Secondary_engine_optimization::PRIMARY_TENTATIVELY);
assert(!m_lex->unit->is_executed());
thd->clear_error();
if (err_seen == ER_PREPARE_FOR_SECONDARY_ENGINE) {
thd->set_secondary_engine_optimization(
Secondary_engine_optimization::SECONDARY);
MYSQL_SET_PS_SECONDARY_ENGINE(m_prepared_stmt, true);
} else {
thd->set_secondary_engine_optimization(
Secondary_engine_optimization::PRIMARY_ONLY);
MYSQL_SET_PS_SECONDARY_ENGINE(m_prepared_stmt, false);
}
// Disable the general log. The query was written to the general log in
// the first attempt to execute it. No need to write it twice.
general_log_temporarily_disabled |= disable_general_log(thd);
error = reprepare(thd);
}
// If (re-?)preparation or optimization failed and it was for
// a secondary storage engine, disable the secondary storage
// engine and try again without it.
if (error && m_lex->m_sql_cmd != nullptr &&
thd->secondary_engine_optimization() ==
Secondary_engine_optimization::SECONDARY &&
!m_lex->unit->is_executed()) {
if (!thd->is_secondary_engine_forced()) {
thd->clear_error();
thd->set_secondary_engine_optimization(
Secondary_engine_optimization::PRIMARY_ONLY);
MYSQL_SET_PS_SECONDARY_ENGINE(m_prepared_stmt, false);
error = reprepare(thd);
if (!error) {
// The reprepared statement should not use a secondary engine.
assert(!m_lex->m_sql_cmd->using_secondary_storage_engine());
m_lex->m_sql_cmd->disable_secondary_storage_engine();
}
}
}
}
if (!error) /* Success */
goto reexecute;
}
reset_stmt_parameters(this);
m_first_execution = false;
// Re-enable the general log if it was temporarily disabled while repreparing
// and executing a statement for a secondary engine.
if (general_log_temporarily_disabled)
thd->variables.option_bits &= ~OPTION_LOG_OFF;
return error;
}
bool Prepared_statement::execute_server_runnable(
THD *thd, Server_runnable *server_runnable) {
Query_arena arena_backup;
bool error;
Query_arena *save_stmt_arena = thd->stmt_arena;
Item_change_list save_change_list;
thd->change_list.move_elements_to(&save_change_list);
m_arena.set_state(Query_arena::STMT_REGULAR_EXECUTION);
assert(m_lex == nullptr);
m_lex = new (m_arena.mem_root) st_lex_local;
if (m_lex == nullptr) return true;
Statement_backup stmt_backup;
stmt_backup.set_thd_to_ps(thd, this);
stmt_backup.save_rlb(thd);
thd->swap_query_arena(m_arena, &arena_backup);
thd->stmt_arena = &m_arena;
error = server_runnable->execute_server_code(thd);
thd->cleanup_after_query();
thd->swap_query_arena(arena_backup, &m_arena);
stmt_backup.restore_thd(thd, this);
stmt_backup.restore_rlb(thd);
thd->stmt_arena = save_stmt_arena;
save_change_list.move_elements_to(&thd->change_list);
// Items and memory will be freed in destructor
return error;
}
/**
Reprepare this prepared statement.
Performs a light reset of the Prepared_statement object (resets its MEM_ROOT
and clears its MEM_ROOT allocated members), and calls prepare() on it again.
The resetting of the MEM_ROOT and clearing of the MEM_ROOT allocated members
is performed by a swap operation (swap_prepared_statement()) on this
Prepared_statement and a newly created, intermediate Prepared_statement. This
both clears the data from this object and stores a backup of the original data
in the intermediate object. If the repreparation fails, the original data is
swapped back into this Prepared_statement.
@param thd current thread.
@retval true an error occurred. Possible errors include
incompatibility of new and old result set
metadata
@retval false success, the statement has been reprepared
*/
bool Prepared_statement::reprepare(THD *thd) {
char saved_cur_db_name_buf[NAME_LEN + 1];
LEX_STRING saved_cur_db_name = {saved_cur_db_name_buf,
sizeof(saved_cur_db_name_buf)};
/*
Create an intermediate Prepared_statement object and move the MEM_ROOT
allocated data of the original Prepared_statement into it. Set up a guard
that moves the data back into the original statement if the repreparation
fails.
*/
Prepared_statement copy(thd);
swap_prepared_statement(&copy);
auto copy_guard =
create_scope_guard([&]() { swap_prepared_statement(&copy); });
thd->status_var.com_stmt_reprepare++;
global_aggregated_stats.get_shard(thd->thread_id()).com_stmt_reprepare++;
/*
m_name has been moved to the copy. Allocate it again in the original
statement.
*/
if (copy.m_name.str != nullptr && set_name(copy.m_name)) return true;
bool cur_db_changed;
if (mysql_opt_change_db(thd, copy.m_db, &saved_cur_db_name, true,
&cur_db_changed))
return true;
/*
Suppress sending metadata to the client while repreparing. It was sent
during the initial preparation.
*/
const unsigned saved_sql_prepare = is_sql_prepare();
set_sql_prepare();
m_arena.is_repreparing = true; // To check whether this is a reprepare.
const bool prepare_error =
prepare(thd, copy.m_query_string.str, copy.m_query_string.length,
copy.m_param_array);
set_sql_prepare(saved_sql_prepare);
m_arena.is_repreparing = false;
if (cur_db_changed)
mysql_change_db(thd, to_lex_cstring(saved_cur_db_name), true);
if (prepare_error) return true;
if (validate_metadata(thd, &copy)) return true;
/* Update reprepare count for this prepared statement in P_S table. */
MYSQL_REPREPARE_PS(m_prepared_stmt);
/*
A new parameter array was created by prepare(). Make sure it contains the
same values as the original array.
*/
assert(m_param_count == copy.m_param_count);
swap_parameter_array(m_param_array, copy.m_param_array, m_param_count);
/*
Clear possible warnings during reprepare, it has to be completely
transparent to the user. We use reset_condition_info() since
there were no separate query id issued for re-prepare.
Sic: we can't simply silence warnings during reprepare, because if
it's failed, we need to return all the warnings to the user.
*/
thd->get_stmt_da()->reset_condition_info(thd);
copy_guard.release();
return false;
}
/**
Validate statement result set metadata (if the statement returns
a result set).
Currently we only check that the number of columns of the result
set did not change.
This is a helper method used during re-prepare.
@param thd current thread.
@param copy the re-prepared prepared statement to verify the metadata of
@retval true error, ER_PS_REBIND is reported
@retval false statement return no or compatible metadata
*/
bool Prepared_statement::validate_metadata(THD *thd, Prepared_statement *copy) {
/**
If this is an SQL prepared statement or EXPLAIN,
return false -- the metadata of the original SELECT,
if any, has not been sent to the client.
*/
if (is_sql_prepare() || m_lex->is_explain()) return false;
if (m_lex->query_block->num_visible_fields() !=
copy->m_lex->query_block->num_visible_fields()) {
/** Column counts mismatch, update the client */
thd->server_status |= SERVER_STATUS_METADATA_CHANGED;
}
return false;
}
/**
Swap the MEM_ROOT allocated data of two prepared statements.
This is a private helper that is used as part of statement reprepare. It is
used in the beginning of reprepare() to clear the MEM_ROOT of the statement
before the new preparation, while keeping the data available as some of it is
needed later in the repreparation. It is also used for restoring the original
data from the copy, should the repreparation fail.
The operation is symmetric. It can be used both for saving an original
statement into a backup, and for restoring the original state of the statement
from the backup.
*/
void Prepared_statement::swap_prepared_statement(Prepared_statement *copy) {
Query_arena tmp_arena;
/* Swap memory roots. */
std::swap(m_mem_root, copy->m_mem_root);
/* Swap the arenas */
m_arena.swap_query_arena(copy->m_arena, &tmp_arena);
copy->m_arena.set_query_arena(tmp_arena);
/* Swap the statement attributes */
std::swap(m_lex, copy->m_lex);
std::swap(m_query_string, copy->m_query_string);
/* Swap mem_roots back, they must continue pointing at the m_mem_roots */
std::swap(m_arena.mem_root, copy->m_arena.mem_root);
/*
Swap the old and the new parameters array. The old array
is allocated in the old arena.
*/
std::swap(m_param_array, copy->m_param_array);
std::swap(m_param_count, copy->m_param_count);
/* Don't swap flags: the copy has IS_SQL_PREPARE always set. */
/* std::swap(flags, copy->flags); */
/* Swap names, the old name is allocated in the wrong memory root */
std::swap(m_name, copy->m_name);
/* Ditto */
std::swap(m_db, copy->m_db);
// Need a new cursor-specific query result after repreparation
std::swap(m_cursor_result, copy->m_cursor_result);
std::swap(m_regular_result, copy->m_regular_result);
std::swap(m_aux_result, copy->m_aux_result);
// Need a new cursor, if requested
std::swap(m_cursor, copy->m_cursor);
}
/**
Execute a prepared statement that has already been prepared.
@param thd current thread.
@param expanded_query A query for binlogging which has all parameter
markers ('?') replaced with their actual values.
@param open_cursor True if an attempt to open a cursor should be made.
Currently used only in the binary protocol.
@note
Preconditions:
- Caller must ensure that thd->change_list and thd->item_list are empty;
This function will free them after execution.
Postconditions.
- There are no items in thd->change_list.
- thd->mem_root may contain memory allocated during execution.
@returns false if success, true if error
*/
bool Prepared_statement::execute(THD *thd, String *expanded_query,
bool open_cursor) {
// This flag is required for proper cleanup in the scope guard.
bool status = true;
Query_arena *old_stmt_arena = nullptr;
char saved_cur_db_name_buf[NAME_LEN + 1];
LEX_STRING saved_cur_db_name = {saved_cur_db_name_buf,
sizeof(saved_cur_db_name_buf)};
bool cur_db_changed = false;
Sql_cmd_dml *sql_cmd = m_lex->m_sql_cmd != nullptr &&
m_lex->m_sql_cmd->sql_cmd_type() == SQL_CMD_DML
? down_cast<Sql_cmd_dml *>(m_lex->m_sql_cmd)
: nullptr;
resourcegroups::Resource_group *src_res_grp = nullptr;
resourcegroups::Resource_group *dest_res_grp = nullptr;
MDL_ticket *ticket = nullptr;
MDL_ticket *cur_ticket = nullptr;
auto mgr_ptr = resourcegroups::Resource_group_mgr::instance();
bool resource_group_switched = false;
// Validate preconditions:
assert(thd->change_list.is_empty());
assert(thd->item_list() == nullptr);
thd->status_var.com_stmt_execute++;
global_aggregated_stats.get_shard(thd->thread_id()).com_stmt_execute++;
/*
Reset the diagnostics area.
For regular statements, this would have happened in the parsing
stage.
SQL prepared statements (SQLCOM_EXECUTE) also have a parsing
stage first (where we find out it's EXECUTE ... [USING ...]).
However, ps-protocol prepared statements have no parsing stage for
COM_STMT_EXECUTE before coming here, so we reset the condition info
here. Since diagnostics statements can't be prepared, we don't need
to make an exception for them.
Note that this will also reset any warnings reported during initial
processing of parameter values, e.g when converting string values to
temporal values.
*/
thd->get_stmt_da()->reset_condition_info(thd);
if (open_cursor) {
// Only DML statements may have assigned a cursor.
if (sql_cmd == nullptr) {
my_error(ER_WRONG_ARGUMENTS, MYF(0), "with cursor");
return true;
}
/*
Some statements that return result sets don't support a cursor,
e.g. SELECT ... INTO.
*/
if (sql_cmd->query_result() != nullptr &&
sql_cmd->query_result()->check_supports_cursor()) {
DBUG_PRINT("info", ("Cursor asked for not SELECT stmt"));
return true;
}
/*
For some statements, such as the CALL statement, query_result() is NULL.
For these, no error is returned and no cursor is opened -- the client
library will recognize this case and materialize the result set.
*/
}
// Prevent recursive calls from e.g. stored routines
if (m_in_use) {
my_error(ER_PS_NO_RECURSION, MYF(0));
return true;
}
m_in_use = true;
// If statement was used as cursor in last execution, close it.
if (m_used_as_cursor) close_cursor();
/*
Set up THD for execution of prepared statement:
- THD::lex becomes statement's LEX
- THD::m_query_string becomes statement's query string
- THD::m_rewritten_query becomes statement's rewritten query
- THD::stmt_arena becomes statement's arena (m_item_list, state)
- THD::mem_root remains the current execution mem_root
*/
Statement_backup stmt_backup;
stmt_backup.set_thd_to_ps(thd, this);
stmt_backup.save_rlb(thd);
auto execute_guard = create_scope_guard([&]() {
// In an error situation, cursor may have been left open, close it:
if (status && open_cursor) {
if (m_cursor == nullptr && m_cursor_result != nullptr) {
m_cursor = m_cursor_result->cursor();
}
if (m_cursor != nullptr && m_cursor->is_open()) {
close_cursor();
}
}
/*
Restore the current database (if changed).
Force switching back to the saved current database (if changed),
because it may be NULL. In this case, mysql_change_db() would generate
an error.
*/
if (cur_db_changed)
mysql_change_db(thd, to_lex_cstring(saved_cur_db_name), true);
cleanup_stmt(thd);
m_lex->release_plugins();
if (resource_group_switched)
mgr_ptr->restore_original_resource_group(thd, src_res_grp, dest_res_grp);
thd->resource_group_ctx()->m_switch_resource_group_str[0] = '\0';
if (ticket != nullptr)
mgr_ptr->release_shared_mdl_for_resource_group(thd, ticket);
if (cur_ticket != nullptr)
mgr_ptr->release_shared_mdl_for_resource_group(thd, cur_ticket);
/*
Note that we cannot call restore_thd() here as that would overwrite
the expanded query in THD::m_query_string, which is needed for slow
logging. Use alloc_query() to make sure the query is allocated on
the correct MEM_ROOT, since otherwise THD::m_query_string could end up
as a dangling pointer (i.e. pointer to freed memory) once
the PS MEM_ROOT is freed.
*/
mysql_mutex_lock(&thd->LOCK_thd_data);
thd->lex = stmt_backup.lex();
mysql_mutex_unlock(&thd->LOCK_thd_data);
alloc_query(thd, thd->query().str, thd->query().length);
thd->stmt_arena = old_stmt_arena;
// Restore the original rewritten query.
stmt_backup.restore_rlb(thd);
if (m_arena.get_state() == Query_arena::STMT_PREPARED)
m_arena.set_state(Query_arena::STMT_EXECUTED);
if (!status && m_lex->sql_command == SQLCOM_CALL)
thd->get_protocol()->send_parameters(&m_lex->param_list,
is_sql_prepare());
m_in_use = false;
// Validate postconditions:
assert(thd->change_list.is_empty());
});
/*
Change the current database (if needed).
Force switching, because the database of the prepared statement may be
NULL (prepared statements can be created while no current database
selected).
*/
if (mysql_opt_change_db(thd, m_db, &saved_cur_db_name, true,
&cur_db_changed)) {
return true;
}
// Allocate the expanded query on the THD memroot
if (expanded_query->length() != 0 &&
alloc_query(thd, expanded_query->ptr(), expanded_query->length())) {
my_error(ER_OUTOFMEMORY, MYF(ME_FATALERROR), expanded_query->length());
return true;
}
/*
Make the statement's statement arena available in the THD.
This is required so that the statement's item list is available.
But notice that thd->mem_root is still the caller's mem_root.
*/
old_stmt_arena = thd->stmt_arena;
thd->stmt_arena = &m_arena;
if (m_lex->check_preparation_invalid(thd)) return true;
/*
Set a hint so mysql_execute_command() won't clear the DA *again*,
thereby discarding any conditions we might raise in here
(e.g. "database we prepared with no longer exists", ER_BAD_DB_ERROR).
*/
m_lex->keep_diagnostics = DA_KEEP_PARSE_ERROR;
// Prepare for a new execution
m_lex->clear_execution();
if (open_cursor) {
Query_result_send *new_result = nullptr;
m_lex->safe_to_cache_query = false;
/*
Initialize a Query_result object before opening the cursor, unless
it has already been created for this statement.
Remark that query result object is created depending upon current
protocol, hence we need to ensure that protocol does not change
during the lifetime of a prepared statement.
cf. Prepared_statement::execute_loop().
*/
if (m_cursor_result != nullptr) {
m_active_protocol = thd->get_protocol();
} else if (sql_cmd->may_use_cursor()) {
if (thd->is_classic_protocol())
new_result = new (m_arena.mem_root) Query_fetch_protocol_binary(thd);
else
new_result = new (m_arena.mem_root) Query_result_send();
if (new_result == nullptr) return true; // OOM
m_cursor_result = new_cursor_result(m_arena.mem_root, new_result);
if (m_cursor_result == nullptr) {
::destroy_at(new_result);
return true;
}
// Saved result for proper destruction
m_aux_result = new_result;
// Signal that query result must be prepared on execution
sql_cmd->set_lazy_result();
}
sql_cmd->set_query_result(m_cursor_result);
m_lex->result = m_cursor_result;
if (m_cursor_result == nullptr) {
m_cursor_result = m_lex->unit->query_result();
}
} else {
if (sql_cmd != nullptr) {
if (m_regular_result == nullptr)
m_regular_result = sql_cmd->query_result();
else
sql_cmd->set_query_result(m_regular_result);
m_lex->result = m_regular_result;
}
}
/*
Log COM_STMT_EXECUTE to the general log. Note, that in case of SQL
prepared statements this causes two records to be output:
Query EXECUTE <statement name>
Execute <statement SQL text>
This is considered user-friendly, since in the
second log entry we output values of parameter markers.
Rewriting/password obfuscation:
- Any passwords in the "Execute" line should be substituted with
their hashes, or a notice.
Rewrite first, execution might replace passwords
with hashes in situ without flagging it, and then we'd make
a hash of that hash.
*/
rewrite_query(thd);
log_execute_line(thd);
const char *display_query_string;
int display_query_length;
if (thd->rewritten_query().length()) {
display_query_string = thd->rewritten_query().ptr();
display_query_length = thd->rewritten_query().length();
} else {
display_query_string = thd->query().str;
display_query_length = thd->query().length;
}
mysql_thread_set_info(display_query_string, display_query_length);
thd->binlog_need_explicit_defaults_ts =
m_lex->binlog_need_explicit_defaults_ts;
resource_group_switched = mgr_ptr->switch_resource_group_if_needed(
thd, &src_res_grp, &dest_res_grp, &ticket, &cur_ticket);
status = mysql_execute_command(thd, true);
if (status) return true;
if (open_cursor) {
if (m_cursor == nullptr && m_cursor_result != nullptr) {
m_cursor = m_cursor_result->cursor();
}
/*
Execution was successful. For most queries, a cursor has been created
and must be opened, however for some queries, no cursor is used.
This is possible if some command writes directly to the network,
bypassing Query_result mechanism. An example of such command is
SHOW PRIVILEGES (whose command class inherits from Sql_cmd_show_noplan).
*/
if (m_cursor != nullptr) {
m_used_as_cursor = true;
/*
NOTE: close_thread_tables() has been called in mysql_execute_command(),
so all tables except for the cursor temporary table have been closed.
*/
if (m_cursor->open(thd)) {
return true;
}
}
}
return false;
}
/** Common part of DEALLOCATE PREPARE and mysqld_stmt_close. */
void Prepared_statement::deallocate(THD *thd) {
/* We account deallocate in the same manner as mysqld_stmt_close */
thd->status_var.com_stmt_close++;
global_aggregated_stats.get_shard(thd->thread_id()).com_stmt_close++;
/* Statement map calls delete stmt on erase */
thd->stmt_map.erase(this);
}