mirror of
https://github.com/MariaDB/server.git
synced 2025-01-16 03:52:35 +01:00
3a8e769836
The follow-up patch to check in mtr tests that recompilation of a SP's instruction doesn't lead to eviction of SP from sp_cache. This patch adds the debug keyword 'check_sp_cache_not_invalidated' checked in sp_cache_flush_obsolete. In case this debug keyword is set the macros DBUG_SUICIDE() called to cause test failure. The function sp_cache_flush_obsolete() is called on opening a stored routine. So setting this keyword before second execution of some stored routine that supposed to cause recompilation of SP's statement will guarantee that this stored routine not evicted from sp_cache. Suggested approach has one limitation - the statement CREATE/ALTER/DROP VIEW forces invalidation of the whole sp_cache (by invoking the function sp_cache_invalidate()). So, for those tests (actually, there are very small number of such tests) that create/alter/drop a view before the second execution of some stored routine, the debug keyword 'check_sp_cache_not_invalidated' isn't set. The proposal to add some way a check that a stored routine is not force out from sp_cache on re-parsing a failing statement of a stored routine was done during reiew, that is the reason the proposed change has been formatted as a separate patch.
324 lines
7.1 KiB
C++
324 lines
7.1 KiB
C++
/* Copyright (c) 2002, 2012, Oracle and/or its affiliates. All rights reserved.
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; version 2 of the License.
|
|
|
|
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 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-1335 USA */
|
|
|
|
#include "mariadb.h"
|
|
#include "sql_priv.h"
|
|
#include "unireg.h"
|
|
#ifdef USE_PRAGMA_IMPLEMENTATION
|
|
#pragma implementation
|
|
#endif
|
|
#include "sp_cache.h"
|
|
#include "sp_head.h"
|
|
|
|
static mysql_mutex_t Cversion_lock;
|
|
static ulong volatile Cversion= 1;
|
|
|
|
|
|
/*
|
|
Cache of stored routines.
|
|
*/
|
|
|
|
class sp_cache
|
|
{
|
|
public:
|
|
sp_cache();
|
|
~sp_cache();
|
|
|
|
/**
|
|
Inserts a sp_head object into a hash table.
|
|
|
|
@returns Success status
|
|
@return TRUE Failure
|
|
@return FALSE Success
|
|
*/
|
|
inline bool insert(sp_head *sp)
|
|
{
|
|
return my_hash_insert(&m_hashtable, (const uchar *)sp);
|
|
}
|
|
|
|
inline sp_head *lookup(char *name, size_t namelen)
|
|
{
|
|
return (sp_head *) my_hash_search(&m_hashtable, (const uchar *)name,
|
|
namelen);
|
|
}
|
|
|
|
inline void remove(sp_head *sp)
|
|
{
|
|
my_hash_delete(&m_hashtable, (uchar *)sp);
|
|
}
|
|
|
|
/**
|
|
Remove all elements from a stored routine cache if the current
|
|
number of elements exceeds the argument value.
|
|
|
|
@param[in] upper_limit_for_elements Soft upper limit of elements that
|
|
can be stored in the cache.
|
|
*/
|
|
void enforce_limit(ulong upper_limit_for_elements)
|
|
{
|
|
if (m_hashtable.records > upper_limit_for_elements)
|
|
my_hash_reset(&m_hashtable);
|
|
}
|
|
|
|
private:
|
|
void init();
|
|
void cleanup();
|
|
|
|
/* All routines in this cache */
|
|
HASH m_hashtable;
|
|
}; // class sp_cache
|
|
|
|
#ifdef HAVE_PSI_INTERFACE
|
|
static PSI_mutex_key key_Cversion_lock;
|
|
|
|
static PSI_mutex_info all_sp_cache_mutexes[]=
|
|
{
|
|
{ &key_Cversion_lock, "Cversion_lock", PSI_FLAG_GLOBAL}
|
|
};
|
|
|
|
static void init_sp_cache_psi_keys(void)
|
|
{
|
|
const char* category= "sql";
|
|
int count;
|
|
|
|
if (PSI_server == NULL)
|
|
return;
|
|
|
|
count= array_elements(all_sp_cache_mutexes);
|
|
PSI_server->register_mutex(category, all_sp_cache_mutexes, count);
|
|
}
|
|
#endif
|
|
|
|
/* Initialize the SP caching once at startup */
|
|
|
|
void sp_cache_init()
|
|
{
|
|
#ifdef HAVE_PSI_INTERFACE
|
|
init_sp_cache_psi_keys();
|
|
#endif
|
|
|
|
mysql_mutex_init(key_Cversion_lock, &Cversion_lock, MY_MUTEX_INIT_FAST);
|
|
}
|
|
|
|
|
|
/*
|
|
Clear the cache *cp and set *cp to NULL.
|
|
|
|
SYNOPSIS
|
|
sp_cache_clear()
|
|
cp Pointer to cache to clear
|
|
|
|
NOTE
|
|
This function doesn't invalidate other caches.
|
|
*/
|
|
|
|
void sp_cache_clear(sp_cache **cp)
|
|
{
|
|
sp_cache *c= *cp;
|
|
|
|
if (c)
|
|
{
|
|
delete c;
|
|
*cp= NULL;
|
|
}
|
|
}
|
|
|
|
|
|
void sp_cache_end()
|
|
{
|
|
mysql_mutex_destroy(&Cversion_lock);
|
|
}
|
|
|
|
|
|
/*
|
|
Insert a routine into the cache.
|
|
|
|
SYNOPSIS
|
|
sp_cache_insert()
|
|
cp The cache to put routine into
|
|
sp Routine to insert.
|
|
|
|
TODO: Perhaps it will be more straightforward if in case we returned an
|
|
error from this function when we couldn't allocate sp_cache. (right
|
|
now failure to put routine into cache will cause a 'SP not found'
|
|
error to be reported at some later time)
|
|
*/
|
|
|
|
void sp_cache_insert(sp_cache **cp, sp_head *sp)
|
|
{
|
|
sp_cache *c;
|
|
|
|
if (!(c= *cp))
|
|
{
|
|
if (!(c= new sp_cache()))
|
|
return; // End of memory error
|
|
}
|
|
/* Reading a ulong variable with no lock. */
|
|
sp->set_sp_cache_version(Cversion);
|
|
DBUG_PRINT("info",("sp_cache: inserting: %s", ErrConvDQName(sp).ptr()));
|
|
c->insert(sp);
|
|
*cp= c; // Update *cp if it was NULL
|
|
}
|
|
|
|
|
|
/*
|
|
Look up a routine in the cache.
|
|
SYNOPSIS
|
|
sp_cache_lookup()
|
|
cp Cache to look into
|
|
name Name of rutine to find
|
|
|
|
NOTE
|
|
An obsolete (but not more obsolete then since last
|
|
sp_cache_flush_obsolete call) routine may be returned.
|
|
|
|
RETURN
|
|
The routine or
|
|
NULL if the routine not found.
|
|
*/
|
|
|
|
sp_head *sp_cache_lookup(sp_cache **cp, const Database_qualified_name *name)
|
|
{
|
|
char buf[NAME_LEN * 2 + 2];
|
|
sp_cache *c= *cp;
|
|
if (! c)
|
|
return NULL;
|
|
return c->lookup(buf, name->make_qname(buf, sizeof(buf)));
|
|
}
|
|
|
|
|
|
/*
|
|
Invalidate all routines in all caches.
|
|
|
|
SYNOPSIS
|
|
sp_cache_invalidate()
|
|
|
|
NOTE
|
|
This is called when a VIEW definition is created or modified (and in some
|
|
other contexts). We can't destroy sp_head objects here as one may modify
|
|
VIEW definitions from prelocking-free SPs.
|
|
*/
|
|
|
|
void sp_cache_invalidate()
|
|
{
|
|
DBUG_PRINT("info",("sp_cache: invalidating"));
|
|
thread_safe_increment(Cversion, &Cversion_lock);
|
|
}
|
|
|
|
|
|
/**
|
|
Remove an out-of-date SP from the cache.
|
|
|
|
@param[in] cp Cache to flush
|
|
@param[in] sp SP to remove.
|
|
|
|
@note This invalidates pointers to sp_head objects this thread
|
|
uses. In practice that means don't call this function when
|
|
inside SP'.
|
|
*/
|
|
|
|
void sp_cache_flush_obsolete(sp_cache **cp, sp_head **sp)
|
|
{
|
|
if ((*sp)->sp_cache_version() < Cversion && !(*sp)->is_invoked())
|
|
{
|
|
DBUG_EXECUTE_IF("check_sp_cache_not_invalidated", DBUG_SUICIDE(););
|
|
(*cp)->remove(*sp);
|
|
*sp= NULL;
|
|
}
|
|
}
|
|
|
|
/**
|
|
Return the current global version of the cache.
|
|
*/
|
|
|
|
ulong sp_cache_version()
|
|
{
|
|
return Cversion;
|
|
}
|
|
|
|
|
|
/**
|
|
Enforce that the current number of elements in the cache don't exceed
|
|
the argument value by flushing the cache if necessary.
|
|
|
|
@param[in] c Cache to check
|
|
@param[in] upper_limit_for_elements Soft upper limit for number of sp_head
|
|
objects that can be stored in the cache.
|
|
*/
|
|
void
|
|
sp_cache_enforce_limit(sp_cache *c, ulong upper_limit_for_elements)
|
|
{
|
|
if (c)
|
|
c->enforce_limit(upper_limit_for_elements);
|
|
}
|
|
|
|
/*************************************************************************
|
|
Internal functions
|
|
*************************************************************************/
|
|
|
|
extern "C" uchar *hash_get_key_for_sp_head(const uchar *ptr, size_t *plen,
|
|
my_bool first);
|
|
extern "C" void hash_free_sp_head(void *p);
|
|
|
|
uchar *hash_get_key_for_sp_head(const uchar *ptr, size_t *plen,
|
|
my_bool first)
|
|
{
|
|
sp_head *sp= (sp_head *)ptr;
|
|
*plen= sp->m_qname.length;
|
|
return (uchar*) sp->m_qname.str;
|
|
}
|
|
|
|
|
|
void hash_free_sp_head(void *p)
|
|
{
|
|
sp_head *sp= (sp_head *)p;
|
|
sp_head::destroy(sp);
|
|
}
|
|
|
|
|
|
sp_cache::sp_cache()
|
|
{
|
|
init();
|
|
}
|
|
|
|
|
|
sp_cache::~sp_cache()
|
|
{
|
|
my_hash_free(&m_hashtable);
|
|
}
|
|
|
|
|
|
void
|
|
sp_cache::init()
|
|
{
|
|
my_hash_init(key_memory_sp_cache, &m_hashtable, system_charset_info, 0, 0, 0,
|
|
hash_get_key_for_sp_head, hash_free_sp_head, 0);
|
|
}
|
|
|
|
|
|
void
|
|
sp_cache::cleanup()
|
|
{
|
|
my_hash_free(&m_hashtable);
|
|
}
|
|
|
|
|
|
void Sp_caches::sp_caches_clear()
|
|
{
|
|
sp_cache_clear(&sp_proc_cache);
|
|
sp_cache_clear(&sp_func_cache);
|
|
sp_cache_clear(&sp_package_spec_cache);
|
|
sp_cache_clear(&sp_package_body_cache);
|
|
}
|