Commit 4f85df4b authored by Konstantin Osipov's avatar Konstantin Osipov

Backport of:

------------------------------------------------------------
revno: 2617.68.24
committer: Dmitry Lenev <dlenev@mysql.com>
branch nick: mysql-next-bg-pre2-2
timestamp: Wed 2009-09-16 17:25:29 +0400
message:
  Pre-requisite patch for fixing bug #30977 "Concurrent statement
  using stored function and DROP FUNCTION breaks SBR".

  Added MDL_request for stored routine as member to Sroutine_hash_entry
  in order to be able perform metadata locking for stored routines in
  future (Sroutine_hash_entry is an equivalent of TABLE_LIST class for
  stored routines).
(WL#4284, follow up fixes).
parent 10db8e79
...@@ -318,6 +318,24 @@ void MDL_request::init(enum_mdl_namespace mdl_namespace, ...@@ -318,6 +318,24 @@ void MDL_request::init(enum_mdl_namespace mdl_namespace,
} }
/**
Initialize a lock request using pre-built MDL_key.
@sa MDL_request::init(namespace, db, name, type).
@param key_arg The pre-built MDL key for the request.
@param mdl_type_arg The MDL lock type for the request.
*/
void MDL_request::init(const MDL_key *key_arg,
enum enum_mdl_type mdl_type_arg)
{
key.mdl_key_init(key_arg);
type= mdl_type_arg;
ticket= NULL;
}
/** /**
Allocate and initialize one lock request. Allocate and initialize one lock request.
...@@ -1254,7 +1272,7 @@ void MDL_context::release_ticket(MDL_ticket *ticket) ...@@ -1254,7 +1272,7 @@ void MDL_context::release_ticket(MDL_ticket *ticket)
MDL_lock *lock= ticket->m_lock; MDL_lock *lock= ticket->m_lock;
DBUG_ENTER("release_ticket"); DBUG_ENTER("release_ticket");
DBUG_PRINT("enter", ("db=%s name=%s", lock->key.db_name(), DBUG_PRINT("enter", ("db=%s name=%s", lock->key.db_name(),
lock->key.table_name())); lock->key.name()));
safe_mutex_assert_owner(&LOCK_mdl); safe_mutex_assert_owner(&LOCK_mdl);
...@@ -1526,7 +1544,7 @@ MDL_ticket::set_cached_object(void *cached_object, ...@@ -1526,7 +1544,7 @@ MDL_ticket::set_cached_object(void *cached_object,
{ {
DBUG_ENTER("mdl_set_cached_object"); DBUG_ENTER("mdl_set_cached_object");
DBUG_PRINT("enter", ("db=%s name=%s cached_object=%p", DBUG_PRINT("enter", ("db=%s name=%s cached_object=%p",
m_lock->key.db_name(), m_lock->key.table_name(), m_lock->key.db_name(), m_lock->key.name(),
cached_object)); cached_object));
/* /*
TODO: This assumption works now since we do get_cached_object() TODO: This assumption works now since we do get_cached_object()
......
...@@ -52,9 +52,14 @@ enum enum_mdl_state { MDL_PENDING, MDL_ACQUIRED }; ...@@ -52,9 +52,14 @@ enum enum_mdl_state { MDL_PENDING, MDL_ACQUIRED };
Different types of objects exist in different namespaces Different types of objects exist in different namespaces
- MDL_TABLE is for tables and views. - MDL_TABLE is for tables and views.
- MDL_PROCEDURE is for stored procedures, stored functions and UDFs. - MDL_FUNCTION is for stored functions.
- MDL_PROCEDURE is for stored procedures.
- MDL_TRIGGER is for triggers.
Note that although there isn't metadata locking on triggers,
it's necessary to have a separate namespace for them since
MDL_key is also used outside of the MDL subsystem.
*/ */
enum enum_mdl_namespace { MDL_TABLE=0, MDL_PROCEDURE }; enum enum_mdl_namespace { MDL_TABLE=0, MDL_FUNCTION, MDL_PROCEDURE, MDL_TRIGGER };
/** Maximal length of key for metadata locking subsystem. */ /** Maximal length of key for metadata locking subsystem. */
#define MAX_MDLKEY_LENGTH (1 + NAME_LEN + 1 + NAME_LEN + 1) #define MAX_MDLKEY_LENGTH (1 + NAME_LEN + 1 + NAME_LEN + 1)
...@@ -78,8 +83,11 @@ public: ...@@ -78,8 +83,11 @@ public:
const char *db_name() const { return m_ptr + 1; } const char *db_name() const { return m_ptr + 1; }
uint db_name_length() const { return m_db_name_length; } uint db_name_length() const { return m_db_name_length; }
const char *table_name() const { return m_ptr + m_db_name_length + 2; } const char *name() const { return m_ptr + m_db_name_length + 2; }
uint table_name_length() const { return m_length - m_db_name_length - 3; } uint name_length() const { return m_length - m_db_name_length - 3; }
enum_mdl_namespace mdl_namespace() const
{ return (enum_mdl_namespace)(m_ptr[0]); }
/** /**
Construct a metadata lock key from a triplet (mdl_namespace, database and name). Construct a metadata lock key from a triplet (mdl_namespace, database and name).
...@@ -179,6 +187,7 @@ public: ...@@ -179,6 +187,7 @@ public:
public: public:
void init(enum_mdl_namespace namespace_arg, const char *db_arg, const char *name_arg, void init(enum_mdl_namespace namespace_arg, const char *db_arg, const char *name_arg,
enum_mdl_type mdl_type_arg); enum_mdl_type mdl_type_arg);
void init(const MDL_key *key_arg, enum_mdl_type mdl_type_arg);
/** Set type of lock request. Can be only applied to pending locks. */ /** Set type of lock request. Can be only applied to pending locks. */
inline void set_type(enum_mdl_type type_arg) inline void set_type(enum_mdl_type type_arg)
{ {
......
...@@ -1391,8 +1391,8 @@ extern "C" uchar* sp_sroutine_key(const uchar *ptr, size_t *plen, ...@@ -1391,8 +1391,8 @@ extern "C" uchar* sp_sroutine_key(const uchar *ptr, size_t *plen,
my_bool first) my_bool first)
{ {
Sroutine_hash_entry *rn= (Sroutine_hash_entry *)ptr; Sroutine_hash_entry *rn= (Sroutine_hash_entry *)ptr;
*plen= rn->key.length; *plen= rn->mdl_request.key.length();
return (uchar *)rn->key.str; return (uchar *)rn->mdl_request.key.ptr();
} }
...@@ -1430,23 +1430,19 @@ extern "C" uchar* sp_sroutine_key(const uchar *ptr, size_t *plen, ...@@ -1430,23 +1430,19 @@ extern "C" uchar* sp_sroutine_key(const uchar *ptr, size_t *plen,
*/ */
bool sp_add_used_routine(Query_tables_list *prelocking_ctx, Query_arena *arena, bool sp_add_used_routine(Query_tables_list *prelocking_ctx, Query_arena *arena,
const LEX_STRING *key, TABLE_LIST *belong_to_view) const MDL_key *key, TABLE_LIST *belong_to_view)
{ {
my_hash_init_opt(&prelocking_ctx->sroutines, system_charset_info, my_hash_init_opt(&prelocking_ctx->sroutines, system_charset_info,
Query_tables_list::START_SROUTINES_HASH_SIZE, Query_tables_list::START_SROUTINES_HASH_SIZE,
0, 0, sp_sroutine_key, 0, 0); 0, 0, sp_sroutine_key, 0, 0);
if (!my_hash_search(&prelocking_ctx->sroutines, (uchar *)key->str, if (!my_hash_search(&prelocking_ctx->sroutines, key->ptr(), key->length()))
key->length))
{ {
Sroutine_hash_entry *rn= Sroutine_hash_entry *rn=
(Sroutine_hash_entry *)arena->alloc(sizeof(Sroutine_hash_entry) + (Sroutine_hash_entry *)arena->alloc(sizeof(Sroutine_hash_entry));
key->length + 1);
if (!rn) // OOM. Error will be reported using fatal_error(). if (!rn) // OOM. Error will be reported using fatal_error().
return FALSE; return FALSE;
rn->key.length= key->length; rn->mdl_request.init(key, MDL_SHARED);
rn->key.str= (char *)rn + sizeof(Sroutine_hash_entry);
memcpy(rn->key.str, key->str, key->length + 1);
my_hash_insert(&prelocking_ctx->sroutines, (uchar *)rn); my_hash_insert(&prelocking_ctx->sroutines, (uchar *)rn);
prelocking_ctx->sroutines_list.link_in_list((uchar *)rn, (uchar **)&rn->next); prelocking_ctx->sroutines_list.link_in_list((uchar *)rn, (uchar **)&rn->next);
rn->belong_to_view= belong_to_view; rn->belong_to_view= belong_to_view;
...@@ -1477,8 +1473,9 @@ bool sp_add_used_routine(Query_tables_list *prelocking_ctx, Query_arena *arena, ...@@ -1477,8 +1473,9 @@ bool sp_add_used_routine(Query_tables_list *prelocking_ctx, Query_arena *arena,
void sp_add_used_routine(Query_tables_list *prelocking_ctx, Query_arena *arena, void sp_add_used_routine(Query_tables_list *prelocking_ctx, Query_arena *arena,
sp_name *rt, char rt_type) sp_name *rt, char rt_type)
{ {
rt->set_routine_type(rt_type); MDL_key key((rt_type == TYPE_ENUM_FUNCTION) ? MDL_FUNCTION : MDL_PROCEDURE,
(void)sp_add_used_routine(prelocking_ctx, arena, &rt->m_sroutines_key, 0); rt->m_db.str, rt->m_name.str);
(void)sp_add_used_routine(prelocking_ctx, arena, &key, 0);
prelocking_ctx->sroutines_list_own_last= prelocking_ctx->sroutines_list.next; prelocking_ctx->sroutines_list_own_last= prelocking_ctx->sroutines_list.next;
prelocking_ctx->sroutines_list_own_elements= prelocking_ctx->sroutines_list_own_elements=
prelocking_ctx->sroutines_list.elements; prelocking_ctx->sroutines_list.elements;
...@@ -1535,7 +1532,8 @@ void sp_update_sp_used_routines(HASH *dst, HASH *src) ...@@ -1535,7 +1532,8 @@ void sp_update_sp_used_routines(HASH *dst, HASH *src)
for (uint i=0 ; i < src->records ; i++) for (uint i=0 ; i < src->records ; i++)
{ {
Sroutine_hash_entry *rt= (Sroutine_hash_entry *)my_hash_element(src, i); Sroutine_hash_entry *rt= (Sroutine_hash_entry *)my_hash_element(src, i);
if (!my_hash_search(dst, (uchar *)rt->key.str, rt->key.length)) if (!my_hash_search(dst, (uchar *)rt->mdl_request.key.ptr(),
rt->mdl_request.key.length()))
my_hash_insert(dst, (uchar *)rt); my_hash_insert(dst, (uchar *)rt);
} }
} }
...@@ -1562,8 +1560,8 @@ sp_update_stmt_used_routines(THD *thd, Query_tables_list *prelocking_ctx, ...@@ -1562,8 +1560,8 @@ sp_update_stmt_used_routines(THD *thd, Query_tables_list *prelocking_ctx,
for (uint i=0 ; i < src->records ; i++) for (uint i=0 ; i < src->records ; i++)
{ {
Sroutine_hash_entry *rt= (Sroutine_hash_entry *)my_hash_element(src, i); Sroutine_hash_entry *rt= (Sroutine_hash_entry *)my_hash_element(src, i);
(void)sp_add_used_routine(prelocking_ctx, thd->stmt_arena, &rt->key, (void)sp_add_used_routine(prelocking_ctx, thd->stmt_arena,
belong_to_view); &rt->mdl_request.key, belong_to_view);
} }
} }
...@@ -1587,8 +1585,8 @@ void sp_update_stmt_used_routines(THD *thd, Query_tables_list *prelocking_ctx, ...@@ -1587,8 +1585,8 @@ void sp_update_stmt_used_routines(THD *thd, Query_tables_list *prelocking_ctx,
{ {
for (Sroutine_hash_entry *rt= (Sroutine_hash_entry *)src->first; for (Sroutine_hash_entry *rt= (Sroutine_hash_entry *)src->first;
rt; rt= rt->next) rt; rt= rt->next)
(void)sp_add_used_routine(prelocking_ctx, thd->stmt_arena, &rt->key, (void)sp_add_used_routine(prelocking_ctx, thd->stmt_arena,
belong_to_view); &rt->mdl_request.key, belong_to_view);
} }
......
...@@ -73,9 +73,10 @@ class Sroutine_hash_entry ...@@ -73,9 +73,10 @@ class Sroutine_hash_entry
{ {
public: public:
/** /**
Set key consisting of one-byte routine type and quoted routine name. Metadata lock request for routine.
MDL_key in this request is also used as a key for set.
*/ */
LEX_STRING key; MDL_request mdl_request;
/** /**
Next element in list linking all routines in set. See also comments Next element in list linking all routines in set. See also comments
for LEX::sroutine/sroutine_list and sp_head::m_sroutines. for LEX::sroutine/sroutine_list and sp_head::m_sroutines.
...@@ -96,7 +97,7 @@ public: ...@@ -96,7 +97,7 @@ public:
void sp_add_used_routine(Query_tables_list *prelocking_ctx, Query_arena *arena, void sp_add_used_routine(Query_tables_list *prelocking_ctx, Query_arena *arena,
sp_name *rt, char rt_type); sp_name *rt, char rt_type);
bool sp_add_used_routine(Query_tables_list *prelocking_ctx, Query_arena *arena, bool sp_add_used_routine(Query_tables_list *prelocking_ctx, Query_arena *arena,
const LEX_STRING *key, TABLE_LIST *belong_to_view); const MDL_key *key, TABLE_LIST *belong_to_view);
void sp_remove_not_own_routines(Query_tables_list *prelocking_ctx); void sp_remove_not_own_routines(Query_tables_list *prelocking_ctx);
void sp_update_sp_used_routines(HASH *dst, HASH *src); void sp_update_sp_used_routines(HASH *dst, HASH *src);
void sp_update_stmt_used_routines(THD *thd, Query_tables_list *prelocking_ctx, void sp_update_stmt_used_routines(THD *thd, Query_tables_list *prelocking_ctx,
......
...@@ -382,31 +382,34 @@ error: ...@@ -382,31 +382,34 @@ error:
} }
/* /**
* Create temporary sp_name object from MDL key.
* sp_name
*
*/
sp_name::sp_name(THD *thd, char *key, uint key_len) @note The lifetime of this object is bound to the lifetime of the MDL_key.
This should be fine as sp_name objects created by this constructor
are mainly used for SP-cache lookups.
@param key MDL key containing database and routine name.
@param qname_buff Buffer to be used for storing quoted routine name
(should be at least 2*NAME_LEN+1+1 bytes).
*/
sp_name::sp_name(const MDL_key *key, char *qname_buff)
{ {
m_sroutines_key.str= key; m_db.str= (char*)key->db_name();
m_sroutines_key.length= key_len; m_db.length= key->db_name_length();
m_qname.str= ++key; m_name.str= (char*)key->name();
m_qname.length= key_len - 1; m_name.length= key->name_length();
if ((m_name.str= strchr(m_qname.str, '.'))) m_qname.str= qname_buff;
if (m_db.length)
{ {
m_db.length= m_name.str - key; strxmov(qname_buff, m_db.str, ".", m_name.str, NullS);
m_db.str= strmake_root(thd->mem_root, key, m_db.length); m_qname.length= m_db.length + 1 + m_name.length;
m_name.str++;
m_name.length= m_qname.length - m_db.length - 1;
} }
else else
{ {
m_name.str= m_qname.str; strmov(qname_buff, m_name.str);
m_name.length= m_qname.length; m_qname.length= m_name.length;
m_db.str= 0;
m_db.length= 0;
} }
m_explicit_name= false; m_explicit_name= false;
} }
...@@ -419,12 +422,10 @@ void ...@@ -419,12 +422,10 @@ void
sp_name::init_qname(THD *thd) sp_name::init_qname(THD *thd)
{ {
const uint dot= !!m_db.length; const uint dot= !!m_db.length;
/* m_sroutines format: m_type + [database + dot] + name + nul */ /* m_qname format: [database + dot] + name + '\0' */
m_sroutines_key.length= 1 + m_db.length + dot + m_name.length; m_qname.length= m_db.length + dot + m_name.length;
if (!(m_sroutines_key.str= (char*) thd->alloc(m_sroutines_key.length + 1))) if (!(m_qname.str= (char*) thd->alloc(m_qname.length + 1)))
return; return;
m_qname.length= m_sroutines_key.length - 1;
m_qname.str= m_sroutines_key.str + 1;
sprintf(m_qname.str, "%.*s%.*s%.*s", sprintf(m_qname.str, "%.*s%.*s%.*s",
(int) m_db.length, (m_db.length ? m_db.str : ""), (int) m_db.length, (m_db.length ? m_db.str : ""),
dot, ".", dot, ".",
...@@ -585,9 +586,6 @@ sp_head::init(LEX *lex) ...@@ -585,9 +586,6 @@ sp_head::init(LEX *lex)
m_defstr.str= NULL; m_defstr.str= NULL;
m_defstr.length= 0; m_defstr.length= 0;
m_sroutines_key.str= NULL;
m_sroutines_key.length= 0;
m_return_field_def.charset= NULL; m_return_field_def.charset= NULL;
DBUG_VOID_RETURN; DBUG_VOID_RETURN;
...@@ -617,14 +615,10 @@ sp_head::init_sp_name(THD *thd, sp_name *spname) ...@@ -617,14 +615,10 @@ sp_head::init_sp_name(THD *thd, sp_name *spname)
if (spname->m_qname.length == 0) if (spname->m_qname.length == 0)
spname->init_qname(thd); spname->init_qname(thd);
m_sroutines_key.length= spname->m_sroutines_key.length; m_qname.length= spname->m_qname.length;
m_sroutines_key.str= (char*) memdup_root(thd->mem_root, m_qname.str= (char*) memdup_root(thd->mem_root,
spname->m_sroutines_key.str, spname->m_qname.str,
spname->m_sroutines_key.length + 1); spname->m_qname.length + 1);
m_sroutines_key.str[0]= static_cast<char>(m_type);
m_qname.length= m_sroutines_key.length - 1;
m_qname.str= m_sroutines_key.str + 1;
DBUG_VOID_RETURN; DBUG_VOID_RETURN;
} }
......
...@@ -109,36 +109,21 @@ public: ...@@ -109,36 +109,21 @@ public:
LEX_STRING m_db; LEX_STRING m_db;
LEX_STRING m_name; LEX_STRING m_name;
LEX_STRING m_qname; LEX_STRING m_qname;
/**
Key representing routine in the set of stored routines used by statement.
Consists of 1-byte routine type and m_qname (which usually refences to
same buffer). Note that one must complete initialization of the key by
calling set_routine_type().
*/
LEX_STRING m_sroutines_key;
bool m_explicit_name; /**< Prepend the db name? */ bool m_explicit_name; /**< Prepend the db name? */
sp_name(LEX_STRING db, LEX_STRING name, bool use_explicit_name) sp_name(LEX_STRING db, LEX_STRING name, bool use_explicit_name)
: m_db(db), m_name(name), m_explicit_name(use_explicit_name) : m_db(db), m_name(name), m_explicit_name(use_explicit_name)
{ {
m_qname.str= m_sroutines_key.str= 0; m_qname.str= 0;
m_qname.length= m_sroutines_key.length= 0; m_qname.length= 0;
} }
/** /** Create temporary sp_name object from MDL key. */
Creates temporary sp_name object from key, used mainly sp_name(const MDL_key *key, char *qname_buff);
for SP-cache lookups.
*/
sp_name(THD *thd, char *key, uint key_len);
// Init. the qualified name from the db and name. // Init. the qualified name from the db and name.
void init_qname(THD *thd); // thd for memroot allocation void init_qname(THD *thd); // thd for memroot allocation
void set_routine_type(char type)
{
m_sroutines_key.str[0]= type;
}
~sp_name() ~sp_name()
{} {}
}; };
...@@ -181,12 +166,6 @@ public: ...@@ -181,12 +166,6 @@ public:
ulong m_sql_mode; ///< For SHOW CREATE and execution ulong m_sql_mode; ///< For SHOW CREATE and execution
LEX_STRING m_qname; ///< db.name LEX_STRING m_qname; ///< db.name
bool m_explicit_name; ///< Prepend the db name? */ bool m_explicit_name; ///< Prepend the db name? */
/**
Key representing routine in the set of stored routines used by statement.
[routine_type]db.name
@sa sp_name::m_sroutines_key
*/
LEX_STRING m_sroutines_key;
LEX_STRING m_db; LEX_STRING m_db;
LEX_STRING m_name; LEX_STRING m_name;
LEX_STRING m_params; LEX_STRING m_params;
......
...@@ -3777,17 +3777,18 @@ open_and_process_routine(THD *thd, Query_tables_list *prelocking_ctx, ...@@ -3777,17 +3777,18 @@ open_and_process_routine(THD *thd, Query_tables_list *prelocking_ctx,
Prelocking_strategy *prelocking_strategy, Prelocking_strategy *prelocking_strategy,
bool *need_prelocking) bool *need_prelocking)
{ {
int type= rt->key.str[0];
DBUG_ENTER("open_and_process_routine"); DBUG_ENTER("open_and_process_routine");
switch (type) switch (rt->mdl_request.key.mdl_namespace())
{ {
case TYPE_ENUM_FUNCTION: case MDL_FUNCTION:
case TYPE_ENUM_PROCEDURE: case MDL_PROCEDURE:
{ {
sp_name name(thd, rt->key.str, rt->key.length); char qname_buff[NAME_LEN*2+1+1];
sp_name name(&rt->mdl_request.key, qname_buff);
sp_head *sp; sp_head *sp;
int type= (rt->mdl_request.key.mdl_namespace() == MDL_FUNCTION) ?
TYPE_ENUM_FUNCTION : TYPE_ENUM_PROCEDURE;
if (sp_cache_routine(thd, type, &name, &sp)) if (sp_cache_routine(thd, type, &name, &sp))
DBUG_RETURN(TRUE); DBUG_RETURN(TRUE);
...@@ -3799,7 +3800,7 @@ open_and_process_routine(THD *thd, Query_tables_list *prelocking_ctx, ...@@ -3799,7 +3800,7 @@ open_and_process_routine(THD *thd, Query_tables_list *prelocking_ctx,
} }
} }
break; break;
case TYPE_ENUM_TRIGGER: case MDL_TRIGGER:
break; break;
default: default:
/* Impossible type value. */ /* Impossible type value. */
...@@ -4304,7 +4305,7 @@ handle_routine(THD *thd, Query_tables_list *prelocking_ctx, ...@@ -4304,7 +4305,7 @@ handle_routine(THD *thd, Query_tables_list *prelocking_ctx,
*/ */
if (rt != (Sroutine_hash_entry*)prelocking_ctx->sroutines_list.first || if (rt != (Sroutine_hash_entry*)prelocking_ctx->sroutines_list.first ||
rt->key.str[0] != TYPE_ENUM_PROCEDURE) rt->mdl_request.key.mdl_namespace() != MDL_PROCEDURE)
{ {
*need_prelocking= TRUE; *need_prelocking= TRUE;
sp_update_stmt_used_routines(thd, prelocking_ctx, &sp->m_sroutines, sp_update_stmt_used_routines(thd, prelocking_ctx, &sp->m_sroutines,
...@@ -8302,7 +8303,7 @@ tdc_wait_for_old_versions(THD *thd, MDL_request_list *mdl_requests) ...@@ -8302,7 +8303,7 @@ tdc_wait_for_old_versions(THD *thd, MDL_request_list *mdl_requests)
while ((mdl_request= it++)) while ((mdl_request= it++))
{ {
if ((share= get_cached_table_share(mdl_request->key.db_name(), if ((share= get_cached_table_share(mdl_request->key.db_name(),
mdl_request->key.table_name())) && mdl_request->key.name())) &&
share->version != refresh_version && share->version != refresh_version &&
!share->used_tables.is_empty()) !share->used_tables.is_empty())
break; break;
......
...@@ -2055,17 +2055,21 @@ add_tables_and_routines_for_triggers(THD *thd, ...@@ -2055,17 +2055,21 @@ add_tables_and_routines_for_triggers(THD *thd,
/* We can have only one trigger per action type currently */ /* We can have only one trigger per action type currently */
sp_head *trigger= table_list->table->triggers->bodies[i][j]; sp_head *trigger= table_list->table->triggers->bodies[i][j];
if (trigger && sp_add_used_routine(prelocking_ctx, thd->stmt_arena, if (trigger)
&trigger->m_sroutines_key,
table_list->belong_to_view))
{ {
trigger->add_used_tables_to_table_list(thd, MDL_key key(MDL_TRIGGER, trigger->m_db.str, trigger->m_name.str);
&prelocking_ctx->query_tables_last,
table_list->belong_to_view); if (sp_add_used_routine(prelocking_ctx, thd->stmt_arena,
sp_update_stmt_used_routines(thd, prelocking_ctx, &key, table_list->belong_to_view))
&trigger->m_sroutines, {
table_list->belong_to_view); trigger->add_used_tables_to_table_list(thd,
trigger->propagate_attributes(prelocking_ctx); &prelocking_ctx->query_tables_last,
table_list->belong_to_view);
sp_update_stmt_used_routines(thd, prelocking_ctx,
&trigger->m_sroutines,
table_list->belong_to_view);
trigger->propagate_attributes(prelocking_ctx);
}
} }
} }
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment