/* 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 *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 *list, const char *name, size_t length, NAMED_ILINK **found) { I_List_iterator 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 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), 1); // Can never fail my_free(key_cache); } bool process_key_caches(process_key_cache_t func, void *param) { I_List_iterator 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(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 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); }