mirror of
https://github.com/MariaDB/server.git
synced 2025-01-27 09:14:17 +01:00
367 lines
10 KiB
C++
367 lines
10 KiB
C++
/* Copyright (c) 2002, 2010, 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 "keycaches.h"
|
|
#include "optimizer_costs.h"
|
|
#include "optimizer_defaults.h"
|
|
#include "handler.h"
|
|
#include "sql_class.h"
|
|
|
|
/****************************************************************************
|
|
Named list handling
|
|
****************************************************************************/
|
|
|
|
NAMED_ILIST key_caches;
|
|
NAMED_ILIST rpl_filters;
|
|
NAMED_ILIST linked_optimizer_costs;
|
|
|
|
extern "C" PSI_memory_key key_memory_KEY_CACHE;
|
|
extern PSI_memory_key key_memory_NAMED_ILINK_name;
|
|
|
|
LEX_CSTRING default_base= {STRING_WITH_LEN("default")};
|
|
|
|
/**
|
|
ilink (intrusive list element) with a name
|
|
*/
|
|
class NAMED_ILINK :public ilink
|
|
{
|
|
public:
|
|
const char *name;
|
|
size_t name_length;
|
|
uchar* data;
|
|
|
|
NAMED_ILINK(I_List<NAMED_ILINK> *links, const char *name_arg,
|
|
size_t name_length_arg, uchar* data_arg)
|
|
:name_length(name_length_arg), data(data_arg)
|
|
{
|
|
name= my_strndup(key_memory_NAMED_ILINK_name, name_arg, name_length,
|
|
MYF(MY_WME));
|
|
links->push_back(this);
|
|
}
|
|
inline bool cmp(const char *name_cmp, size_t length)
|
|
{
|
|
return !system_charset_info->strnncoll(name, name_length, name_cmp, length);
|
|
}
|
|
~NAMED_ILINK() override
|
|
{
|
|
my_free((void *) name);
|
|
}
|
|
};
|
|
|
|
uchar* find_named(I_List<NAMED_ILINK> *list, const char *name, size_t length,
|
|
NAMED_ILINK **found)
|
|
{
|
|
I_List_iterator<NAMED_ILINK> it(*list);
|
|
NAMED_ILINK *element;
|
|
while ((element= it++))
|
|
{
|
|
if (element->cmp(name, length))
|
|
{
|
|
if (found)
|
|
*found= element;
|
|
return element->data;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
bool NAMED_ILIST::delete_element(const char *name, size_t length,
|
|
void (*free_element)(const char *name, void*))
|
|
{
|
|
I_List_iterator<NAMED_ILINK> it(*this);
|
|
NAMED_ILINK *element;
|
|
DBUG_ENTER("NAMED_ILIST::delete_element");
|
|
while ((element= it++))
|
|
{
|
|
if (element->cmp(name, length))
|
|
{
|
|
(*free_element)(element->name, element->data);
|
|
delete element;
|
|
DBUG_RETURN(0);
|
|
}
|
|
}
|
|
DBUG_RETURN(1);
|
|
}
|
|
|
|
void NAMED_ILIST::delete_elements(void (*free_element)(const char *name, void*))
|
|
{
|
|
NAMED_ILINK *element;
|
|
DBUG_ENTER("NAMED_ILIST::delete_elements");
|
|
while ((element= get()))
|
|
{
|
|
(*free_element)(element->name, element->data);
|
|
delete element;
|
|
}
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
|
|
/* Key cache functions */
|
|
|
|
KEY_CACHE zero_key_cache; ///< @@nonexistent_cache.param->value_ptr() points here
|
|
|
|
KEY_CACHE *get_key_cache(const LEX_CSTRING *cache_name)
|
|
{
|
|
if (!cache_name || ! cache_name->length)
|
|
cache_name= &default_base;
|
|
return ((KEY_CACHE*) find_named(&key_caches,
|
|
cache_name->str, cache_name->length, 0));
|
|
}
|
|
|
|
KEY_CACHE *create_key_cache(const char *name, size_t length)
|
|
{
|
|
KEY_CACHE *key_cache;
|
|
DBUG_ENTER("create_key_cache");
|
|
DBUG_PRINT("enter",("name: %.*s", (int)length, name));
|
|
|
|
if ((key_cache= (KEY_CACHE*) my_malloc(key_memory_KEY_CACHE,
|
|
sizeof(KEY_CACHE), MYF(MY_ZEROFILL | MY_WME))))
|
|
{
|
|
if (!new NAMED_ILINK(&key_caches, name, length, (uchar*) key_cache))
|
|
{
|
|
my_free(key_cache);
|
|
key_cache= 0;
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
Set default values for a key cache
|
|
The values in dflt_key_cache_var is set by my_getopt() at startup
|
|
|
|
We don't set 'buff_size' as this is used to enable the key cache
|
|
*/
|
|
key_cache->param_block_size= dflt_key_cache_var.param_block_size;
|
|
key_cache->param_division_limit= dflt_key_cache_var.param_division_limit;
|
|
key_cache->param_age_threshold= dflt_key_cache_var.param_age_threshold;
|
|
key_cache->param_partitions= dflt_key_cache_var.param_partitions;
|
|
}
|
|
}
|
|
DBUG_RETURN(key_cache);
|
|
}
|
|
|
|
|
|
KEY_CACHE *get_or_create_key_cache(const char *name, size_t length)
|
|
{
|
|
LEX_CSTRING key_cache_name;
|
|
KEY_CACHE *key_cache;
|
|
|
|
key_cache_name.str= name;
|
|
key_cache_name.length= length;
|
|
if (!(key_cache= get_key_cache(&key_cache_name)))
|
|
key_cache= create_key_cache(name, length);
|
|
return key_cache;
|
|
}
|
|
|
|
|
|
void free_key_cache(const char *name, void *key_cache)
|
|
{
|
|
end_key_cache(static_cast<KEY_CACHE *>(key_cache), 1); // Can never fail
|
|
my_free(key_cache);
|
|
}
|
|
|
|
|
|
bool process_key_caches(process_key_cache_t func, void *param)
|
|
{
|
|
I_List_iterator<NAMED_ILINK> it(key_caches);
|
|
NAMED_ILINK *element;
|
|
int res= 0;
|
|
|
|
while ((element= it++))
|
|
{
|
|
KEY_CACHE *key_cache= (KEY_CACHE *) element->data;
|
|
res |= func(element->name, key_cache, param);
|
|
}
|
|
return res != 0;
|
|
}
|
|
|
|
/* Rpl_filter functions */
|
|
|
|
LEX_CSTRING default_rpl_filter_base= {STRING_WITH_LEN("")};
|
|
|
|
Rpl_filter *get_rpl_filter(LEX_CSTRING *filter_name)
|
|
{
|
|
if (!filter_name->length)
|
|
filter_name= &default_rpl_filter_base;
|
|
return ((Rpl_filter*) find_named(&rpl_filters,
|
|
filter_name->str, filter_name->length, 0));
|
|
}
|
|
|
|
Rpl_filter *create_rpl_filter(const char *name, size_t length)
|
|
{
|
|
Rpl_filter *filter;
|
|
DBUG_ENTER("create_rpl_filter");
|
|
DBUG_PRINT("enter",("name: %.*s", (int)length, name));
|
|
|
|
filter= new Rpl_filter;
|
|
if (filter)
|
|
{
|
|
if (!new NAMED_ILINK(&rpl_filters, name, length, (uchar*) filter))
|
|
{
|
|
delete filter;
|
|
filter= 0;
|
|
}
|
|
}
|
|
DBUG_RETURN(filter);
|
|
}
|
|
|
|
|
|
Rpl_filter *get_or_create_rpl_filter(const char *name, size_t length)
|
|
{
|
|
LEX_CSTRING rpl_filter_name;
|
|
Rpl_filter *filter;
|
|
|
|
rpl_filter_name.str= (char *) name;
|
|
rpl_filter_name.length= length;
|
|
if (!(filter= get_rpl_filter(&rpl_filter_name)))
|
|
filter= create_rpl_filter(name, length);
|
|
return filter;
|
|
}
|
|
|
|
void free_rpl_filter(const char *name, void *filter)
|
|
{
|
|
delete static_cast<Rpl_filter*>(filter);
|
|
}
|
|
|
|
void free_all_rpl_filters()
|
|
{
|
|
rpl_filters.delete_elements(free_rpl_filter);
|
|
}
|
|
|
|
|
|
/******************************************************************************
|
|
Optimizer costs functions
|
|
******************************************************************************/
|
|
|
|
LEX_CSTRING default_costs_base= {STRING_WITH_LEN("default")};
|
|
|
|
OPTIMIZER_COSTS default_optimizer_costs=
|
|
{
|
|
DEFAULT_DISK_READ_COST, // disk_read_cost
|
|
DEFAULT_INDEX_BLOCK_COPY_COST, // index_block_copy_cost
|
|
DEFAULT_WHERE_COST/4, // key_cmp_cost
|
|
DEFAULT_KEY_COPY_COST, // key_copy_cost
|
|
DEFAULT_KEY_LOOKUP_COST, // key_lookup_cost
|
|
DEFAULT_KEY_NEXT_FIND_COST, // key_next_find_cost
|
|
DEFAULT_DISK_READ_RATIO, // disk_read_ratio
|
|
DEFAULT_ROW_COPY_COST, // row_copy_cost
|
|
DEFAULT_ROW_LOOKUP_COST, // row_lookup_cost
|
|
DEFAULT_ROW_NEXT_FIND_COST, // row_next_find_cost
|
|
DEFAULT_ROWID_COMPARE_COST, // rowid_compare_cost
|
|
DEFAULT_ROWID_COPY_COST, // rowid_copy_cost
|
|
1 // Cannot be deleted
|
|
};
|
|
|
|
OPTIMIZER_COSTS heap_optimizer_costs, tmp_table_optimizer_costs;
|
|
|
|
OPTIMIZER_COSTS *get_optimizer_costs(const LEX_CSTRING *cache_name)
|
|
{
|
|
if (!cache_name->length)
|
|
return &default_optimizer_costs;
|
|
return ((OPTIMIZER_COSTS*) find_named(&linked_optimizer_costs,
|
|
cache_name->str, cache_name->length,
|
|
0));
|
|
}
|
|
|
|
OPTIMIZER_COSTS *create_optimizer_costs(const char *name, size_t length)
|
|
{
|
|
OPTIMIZER_COSTS *optimizer_costs;
|
|
DBUG_ENTER("create_optimizer_costs");
|
|
DBUG_PRINT("enter",("name: %.*s", (int) length, name));
|
|
|
|
if ((optimizer_costs= (OPTIMIZER_COSTS*)
|
|
my_malloc(key_memory_KEY_CACHE,
|
|
sizeof(OPTIMIZER_COSTS), MYF(MY_ZEROFILL | MY_WME))))
|
|
{
|
|
if (!new NAMED_ILINK(&linked_optimizer_costs, name, length,
|
|
(uchar*) optimizer_costs))
|
|
{
|
|
my_free(optimizer_costs);
|
|
optimizer_costs= 0;
|
|
}
|
|
else
|
|
{
|
|
/* Mark that values are not yet set */
|
|
for (uint i=0 ; i < sizeof(OPTIMIZER_COSTS)/sizeof(double) ; i++)
|
|
((double*) optimizer_costs)[i]= OPTIMIZER_COST_UNDEF;
|
|
}
|
|
}
|
|
DBUG_RETURN(optimizer_costs);
|
|
}
|
|
|
|
|
|
OPTIMIZER_COSTS *get_or_create_optimizer_costs(const char *name, size_t length)
|
|
{
|
|
LEX_CSTRING optimizer_costs_name;
|
|
OPTIMIZER_COSTS *optimizer_costs;
|
|
|
|
optimizer_costs_name.str= name;
|
|
optimizer_costs_name.length= length;
|
|
if (!(optimizer_costs= get_optimizer_costs(&optimizer_costs_name)))
|
|
optimizer_costs= create_optimizer_costs(name, length);
|
|
return optimizer_costs;
|
|
}
|
|
|
|
extern "C"
|
|
{
|
|
bool process_optimizer_costs(process_optimizer_costs_t func, TABLE *param)
|
|
{
|
|
I_List_iterator<NAMED_ILINK> it(linked_optimizer_costs);
|
|
NAMED_ILINK *element;
|
|
int res= 0;
|
|
|
|
while ((element= it++))
|
|
{
|
|
LEX_CSTRING name= { element->name, element->name_length };
|
|
OPTIMIZER_COSTS *costs= (OPTIMIZER_COSTS *) element->data;
|
|
res |= func(&name, costs, param);
|
|
}
|
|
return res != 0;
|
|
}
|
|
}
|
|
|
|
bool create_default_optimizer_costs()
|
|
{
|
|
return (new NAMED_ILINK(&linked_optimizer_costs,
|
|
default_base.str, default_base.length,
|
|
(uchar*) &default_optimizer_costs)) == 0;
|
|
}
|
|
|
|
|
|
/*
|
|
Make a copy of heap and tmp_table engine costs to be able to create
|
|
internal temporary tables without taking a mutex.
|
|
*/
|
|
|
|
void copy_tmptable_optimizer_costs()
|
|
{
|
|
memcpy(&heap_optimizer_costs, heap_hton->optimizer_costs,
|
|
sizeof(heap_optimizer_costs));
|
|
memcpy(&tmp_table_optimizer_costs, TMP_ENGINE_HTON->optimizer_costs,
|
|
sizeof(tmp_table_optimizer_costs));
|
|
}
|
|
|
|
|
|
static void free_optimizer_costs(const char *name, void *cost)
|
|
{
|
|
if ((OPTIMIZER_COSTS*) cost != &default_optimizer_costs)
|
|
my_free(cost);
|
|
}
|
|
|
|
void free_all_optimizer_costs()
|
|
{
|
|
linked_optimizer_costs.delete_elements(free_optimizer_costs);
|
|
}
|