2020-01-26 20:27:13 +04:00
|
|
|
/* Copyright (C) 2010, 2020, MariaDB Corporation.
|
2010-04-08 14:10:05 +02:00
|
|
|
|
|
|
|
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
|
Update FSF address
This commit is based on the work of Michal Schorm, rebased on the
earliest MariaDB version.
Th command line used to generate this diff was:
find ./ -type f \
-exec sed -i -e 's/Foundation, Inc., 59 Temple Place, Suite 330, Boston, /Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, /g' {} \; \
-exec sed -i -e 's/Foundation, Inc. 59 Temple Place.* Suite 330, Boston, /Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, /g' {} \; \
-exec sed -i -e 's/MA.*.....-1307.*USA/MA 02110-1335 USA/g' {} \; \
-exec sed -i -e 's/Foundation, Inc., 59 Temple/Foundation, Inc., 51 Franklin/g' {} \; \
-exec sed -i -e 's/Place, Suite 330, Boston, MA.*02111-1307.*USA/Street, Fifth Floor, Boston, MA 02110-1335 USA/g' {} \; \
-exec sed -i -e 's/MA.*.....-1307/MA 02110-1335/g' {} \;
2019-05-10 20:49:46 +03:00
|
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1335 USA */
|
2010-04-08 14:10:05 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
@file
|
|
|
|
|
|
|
|
Engine defined options of tables/fields/keys in CREATE/ALTER TABLE.
|
|
|
|
*/
|
|
|
|
|
2017-06-18 06:42:16 +03:00
|
|
|
#include "mariadb.h"
|
2010-04-08 14:10:05 +02:00
|
|
|
#include "create_options.h"
|
|
|
|
#include <my_getopt.h>
|
2013-04-09 23:27:24 +02:00
|
|
|
#include "set_var.h"
|
2010-04-08 14:10:05 +02:00
|
|
|
|
2017-03-01 08:27:39 +02:00
|
|
|
#define FRM_QUOTED_VALUE 0x8000U
|
2010-04-08 14:10:05 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
Links this item to the given list end
|
|
|
|
|
|
|
|
@param start The list beginning or NULL
|
|
|
|
@param end The list last element or does not matter
|
|
|
|
*/
|
|
|
|
|
|
|
|
void engine_option_value::link(engine_option_value **start,
|
|
|
|
engine_option_value **end)
|
|
|
|
{
|
|
|
|
DBUG_ENTER("engine_option_value::link");
|
|
|
|
DBUG_PRINT("enter", ("name: '%s' (%u) value: '%s' (%u)",
|
|
|
|
name.str, (uint) name.length,
|
|
|
|
value.str, (uint) value.length));
|
|
|
|
engine_option_value *opt;
|
|
|
|
/* check duplicates to avoid writing them to frm*/
|
|
|
|
for(opt= *start;
|
|
|
|
opt && ((opt->parsed && !opt->value.str) ||
|
2020-01-26 20:27:13 +04:00
|
|
|
system_charset_info->strnncoll(name.str, name.length,
|
|
|
|
opt->name.str, opt->name.length));
|
2010-04-08 14:10:05 +02:00
|
|
|
opt= opt->next) /* no-op */;
|
|
|
|
if (opt)
|
|
|
|
{
|
|
|
|
opt->value.str= NULL; /* remove previous value */
|
|
|
|
opt->parsed= TRUE; /* and don't issue warnings for it anymore */
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
Add this option to the end of the list
|
|
|
|
|
|
|
|
@note: We add even if it is opt->value.str == NULL because it can be
|
|
|
|
ALTER TABLE to remove the option.
|
|
|
|
*/
|
|
|
|
if (*start)
|
|
|
|
{
|
|
|
|
(*end)->next= this;
|
|
|
|
*end= this;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
note that is *start == 0, the value of *end does not matter,
|
|
|
|
it can be uninitialized.
|
|
|
|
*/
|
|
|
|
*start= *end= this;
|
|
|
|
}
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool report_wrong_value(THD *thd, const char *name, const char *val,
|
2013-04-09 23:27:14 +02:00
|
|
|
bool suppress_warning)
|
2010-04-08 14:10:05 +02:00
|
|
|
{
|
2010-05-12 19:56:05 +02:00
|
|
|
if (suppress_warning)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (!(thd->variables.sql_mode & MODE_IGNORE_BAD_TABLE_OPTIONS) &&
|
|
|
|
!thd->slave_thread)
|
2010-04-08 14:10:05 +02:00
|
|
|
{
|
|
|
|
my_error(ER_BAD_OPTION_VALUE, MYF(0), val, name);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-06-15 18:32:08 +03:00
|
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN, ER_BAD_OPTION_VALUE,
|
2015-07-06 20:24:14 +03:00
|
|
|
ER_THD(thd, ER_BAD_OPTION_VALUE), val, name);
|
2010-04-08 14:10:05 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool report_unknown_option(THD *thd, engine_option_value *val,
|
2013-04-09 23:27:14 +02:00
|
|
|
bool suppress_warning)
|
2010-04-08 14:10:05 +02:00
|
|
|
{
|
|
|
|
DBUG_ENTER("report_unknown_option");
|
2010-05-12 19:56:05 +02:00
|
|
|
|
2021-01-10 01:31:38 +01:00
|
|
|
if (val->parsed || suppress_warning || thd->slave_thread)
|
2010-04-08 14:10:05 +02:00
|
|
|
{
|
|
|
|
DBUG_PRINT("info", ("parsed => exiting"));
|
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
|
|
|
|
2021-01-10 01:31:38 +01:00
|
|
|
if (!(thd->variables.sql_mode & MODE_IGNORE_BAD_TABLE_OPTIONS))
|
2010-04-08 14:10:05 +02:00
|
|
|
{
|
|
|
|
my_error(ER_UNKNOWN_OPTION, MYF(0), val->name.str);
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
|
|
|
|
2013-06-15 18:32:08 +03:00
|
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN,
|
2015-07-06 20:24:14 +03:00
|
|
|
ER_UNKNOWN_OPTION, ER_THD(thd, ER_UNKNOWN_OPTION),
|
|
|
|
val->name.str);
|
2010-04-08 14:10:05 +02:00
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
|
|
|
|
2013-09-25 19:42:22 +02:00
|
|
|
#define value_ptr(STRUCT,OPT) ((char*)(STRUCT) + (OPT)->offset)
|
|
|
|
|
2024-10-17 15:57:03 +02:00
|
|
|
static bool set_one_value(ha_create_table_option *opt, THD *thd,
|
|
|
|
const LEX_CSTRING *value, void *base,
|
|
|
|
bool suppress_warning, MEM_ROOT *root)
|
2010-04-08 14:10:05 +02:00
|
|
|
{
|
|
|
|
DBUG_ENTER("set_one_value");
|
2017-09-19 17:45:17 +00:00
|
|
|
DBUG_PRINT("enter", ("opt: %p type: %u name '%s' value: '%s'",
|
2024-10-17 15:57:03 +02:00
|
|
|
opt, opt->type, opt->name,
|
2010-04-08 14:10:05 +02:00
|
|
|
(value->str ? value->str : "<DEFAULT>")));
|
|
|
|
switch (opt->type)
|
|
|
|
{
|
2013-04-09 23:27:24 +02:00
|
|
|
case HA_OPTION_TYPE_SYSVAR:
|
2019-08-30 12:51:37 +03:00
|
|
|
// HA_OPTION_TYPE_SYSVAR's are replaced in resolve_sysvars()
|
|
|
|
break; // to DBUG_ASSERT(0)
|
2010-04-08 14:10:05 +02:00
|
|
|
case HA_OPTION_TYPE_ULL:
|
|
|
|
{
|
2013-09-25 19:42:22 +02:00
|
|
|
ulonglong *val= (ulonglong*)value_ptr(base, opt);
|
2010-04-08 14:10:05 +02:00
|
|
|
if (!value->str)
|
|
|
|
{
|
|
|
|
*val= opt->def_value;
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
2024-10-17 15:57:03 +02:00
|
|
|
my_option optp= { opt->name, 1, 0, (uchar **)val, 0, 0, GET_ULL,
|
2012-09-26 18:29:49 +02:00
|
|
|
REQUIRED_ARG, (longlong)opt->def_value, (longlong)opt->min_value,
|
2024-10-17 15:57:03 +02:00
|
|
|
opt->max_value, 0, (long) opt->block_size, 0 };
|
2010-04-08 14:10:05 +02:00
|
|
|
|
|
|
|
ulonglong orig_val= strtoull(value->str, NULL, 10);
|
|
|
|
my_bool unused;
|
|
|
|
*val= orig_val;
|
|
|
|
*val= getopt_ull_limit_value(*val, &optp, &unused);
|
|
|
|
if (*val == orig_val)
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
|
|
|
DBUG_RETURN(report_wrong_value(thd, opt->name, value->str,
|
|
|
|
suppress_warning));
|
|
|
|
}
|
|
|
|
case HA_OPTION_TYPE_STRING:
|
|
|
|
{
|
2013-09-25 19:42:22 +02:00
|
|
|
char **val= (char **)value_ptr(base, opt);
|
2010-04-08 14:10:05 +02:00
|
|
|
if (!value->str)
|
|
|
|
{
|
|
|
|
*val= 0;
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(*val= strmake_root(root, value->str, value->length)))
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
|
|
|
case HA_OPTION_TYPE_ENUM:
|
|
|
|
{
|
2013-09-25 19:42:22 +02:00
|
|
|
uint *val= (uint *)value_ptr(base, opt), num;
|
2010-04-08 14:10:05 +02:00
|
|
|
|
2010-08-07 15:27:23 +03:00
|
|
|
*val= (uint) opt->def_value;
|
2010-04-08 14:10:05 +02:00
|
|
|
if (!value->str)
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
|
|
|
const char *start= opt->values, *end;
|
|
|
|
|
|
|
|
num= 0;
|
|
|
|
while (*start)
|
|
|
|
{
|
|
|
|
for (end=start;
|
|
|
|
*end && *end != ',';
|
2016-05-17 15:27:10 +04:00
|
|
|
end++) /* no-op */;
|
2020-01-26 20:27:13 +04:00
|
|
|
if (!system_charset_info->strnncoll(start, end-start,
|
|
|
|
value->str, value->length))
|
2010-04-08 14:10:05 +02:00
|
|
|
{
|
|
|
|
*val= num;
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
}
|
2010-11-30 23:11:03 +02:00
|
|
|
if (*end)
|
|
|
|
end++;
|
2010-04-08 14:10:05 +02:00
|
|
|
start= end;
|
|
|
|
num++;
|
|
|
|
}
|
|
|
|
|
|
|
|
DBUG_RETURN(report_wrong_value(thd, opt->name, value->str,
|
|
|
|
suppress_warning));
|
|
|
|
}
|
|
|
|
case HA_OPTION_TYPE_BOOL:
|
|
|
|
{
|
2013-09-25 19:42:22 +02:00
|
|
|
bool *val= (bool *)value_ptr(base, opt);
|
2010-04-08 14:10:05 +02:00
|
|
|
*val= opt->def_value;
|
|
|
|
|
|
|
|
if (!value->str)
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
|
2020-01-26 20:27:13 +04:00
|
|
|
if (!system_charset_info->strnncoll("NO", 2, value->str, value->length) ||
|
|
|
|
!system_charset_info->strnncoll("OFF", 3, value->str, value->length) ||
|
|
|
|
!system_charset_info->strnncoll("0", 1, value->str, value->length))
|
2010-04-08 14:10:05 +02:00
|
|
|
{
|
|
|
|
*val= FALSE;
|
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
|
|
|
|
2020-01-26 20:27:13 +04:00
|
|
|
if (!system_charset_info->strnncoll("YES", 3, value->str, value->length) ||
|
|
|
|
!system_charset_info->strnncoll("ON", 2, value->str, value->length) ||
|
|
|
|
!system_charset_info->strnncoll("1", 1, value->str, value->length))
|
2010-04-08 14:10:05 +02:00
|
|
|
{
|
|
|
|
*val= TRUE;
|
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
DBUG_RETURN(report_wrong_value(thd, opt->name, value->str,
|
|
|
|
suppress_warning));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
DBUG_ASSERT(0);
|
|
|
|
my_error(ER_UNKNOWN_ERROR, MYF(0));
|
|
|
|
DBUG_RETURN(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const size_t ha_option_type_sizeof[]=
|
|
|
|
{ sizeof(ulonglong), sizeof(char *), sizeof(uint), sizeof(bool)};
|
|
|
|
|
2024-10-17 15:57:03 +02:00
|
|
|
/**
|
|
|
|
Appends values of sysvar-based options if needed
|
|
|
|
|
|
|
|
@param thd thread handler
|
|
|
|
@param option_list list of options given by user
|
|
|
|
@param rules list of option description by engine
|
|
|
|
@param root MEM_ROOT where allocate memory
|
|
|
|
|
|
|
|
@retval TRUE Error
|
|
|
|
@retval FALSE OK
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool extend_option_list(THD* thd, handlerton *hton, bool create,
|
|
|
|
engine_option_value **option_list,
|
2024-10-23 14:43:32 +02:00
|
|
|
ha_create_table_option *rules)
|
2024-10-17 15:57:03 +02:00
|
|
|
{
|
|
|
|
DBUG_ENTER("extend_option_list");
|
2024-10-23 14:43:32 +02:00
|
|
|
MEM_ROOT *root= thd->mem_root;
|
|
|
|
bool extended= false;
|
2024-10-17 15:57:03 +02:00
|
|
|
|
|
|
|
for (ha_create_table_option *opt= rules; rules && opt->name; opt++)
|
|
|
|
{
|
|
|
|
if (opt->var)
|
|
|
|
{
|
|
|
|
engine_option_value *found= NULL, *last;
|
|
|
|
for (engine_option_value *val= *option_list; val; val= val->next)
|
|
|
|
{
|
|
|
|
last= val;
|
|
|
|
if (!system_charset_info->strnncoll(opt->name, opt->name_length,
|
|
|
|
val->name.str, val->name.length))
|
|
|
|
found= val; // find the last matching
|
|
|
|
}
|
|
|
|
if (found ? !found->value.str : create)
|
|
|
|
{
|
|
|
|
/* add the current value of the corresponding sysvar to the list */
|
|
|
|
sys_var *sysvar= find_hton_sysvar(hton, opt->var);
|
|
|
|
DBUG_ASSERT(sysvar);
|
|
|
|
|
|
|
|
if (!sysvar->session_is_default(thd))
|
|
|
|
{
|
|
|
|
StringBuffer<256> sbuf(system_charset_info);
|
|
|
|
String *str= sysvar->val_str(&sbuf, thd, OPT_SESSION, &null_clex_str);
|
|
|
|
DBUG_ASSERT(str);
|
|
|
|
|
|
|
|
LEX_CSTRING name= { opt->name, opt->name_length };
|
|
|
|
LEX_CSTRING value= safe_lexcstrdup_root(root, str->to_lex_cstring());
|
|
|
|
if (found)
|
|
|
|
found->value= value;
|
|
|
|
else
|
2024-10-23 14:43:32 +02:00
|
|
|
{
|
|
|
|
if (!extended)
|
|
|
|
{
|
2024-10-28 10:29:27 +01:00
|
|
|
if (*option_list)
|
|
|
|
thd->register_item_tree_change((Item**)&(last->next));
|
2024-10-23 14:43:32 +02:00
|
|
|
extended= true;
|
|
|
|
}
|
2024-10-17 15:57:03 +02:00
|
|
|
new (root) engine_option_value(name, value,
|
|
|
|
opt->type != HA_OPTION_TYPE_ULL, option_list, &last);
|
2024-10-23 14:43:32 +02:00
|
|
|
}
|
2024-10-17 15:57:03 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-04-08 14:10:05 +02:00
|
|
|
/**
|
|
|
|
Creates option structure and parses list of options in it
|
|
|
|
|
|
|
|
@param thd thread handler
|
|
|
|
@param option_struct where to store pointer on the option struct
|
|
|
|
@param option_list list of options given by user
|
|
|
|
@param rules list of option description by engine
|
|
|
|
@param suppress_warning second parse so we do not need warnings
|
|
|
|
@param root MEM_ROOT where allocate memory
|
|
|
|
|
|
|
|
@retval TRUE Error
|
|
|
|
@retval FALSE OK
|
|
|
|
*/
|
|
|
|
|
2024-10-17 15:57:03 +02:00
|
|
|
bool parse_option_list(THD* thd, void *option_struct_arg,
|
2013-04-09 23:27:24 +02:00
|
|
|
engine_option_value **option_list,
|
2013-04-09 23:27:14 +02:00
|
|
|
ha_create_table_option *rules,
|
|
|
|
bool suppress_warning, MEM_ROOT *root)
|
2010-04-08 14:10:05 +02:00
|
|
|
{
|
|
|
|
ha_create_table_option *opt;
|
|
|
|
size_t option_struct_size= 0;
|
2013-04-09 23:27:24 +02:00
|
|
|
engine_option_value *val, *last;
|
2011-05-10 18:19:11 +02:00
|
|
|
void **option_struct= (void**)option_struct_arg;
|
2010-04-08 14:10:05 +02:00
|
|
|
DBUG_ENTER("parse_option_list");
|
|
|
|
DBUG_PRINT("enter",
|
2013-04-09 23:27:14 +02:00
|
|
|
("struct: %p list: %p rules: %p suppress_warning: %u root: %p",
|
2013-04-09 23:27:24 +02:00
|
|
|
*option_struct, *option_list, rules,
|
2013-04-09 23:27:14 +02:00
|
|
|
(uint) suppress_warning, root));
|
2010-04-08 14:10:05 +02:00
|
|
|
|
|
|
|
if (rules)
|
|
|
|
{
|
|
|
|
for (opt= rules; opt->name; opt++)
|
|
|
|
set_if_bigger(option_struct_size, opt->offset +
|
|
|
|
ha_option_type_sizeof[opt->type]);
|
|
|
|
|
|
|
|
*option_struct= alloc_root(root, option_struct_size);
|
|
|
|
}
|
|
|
|
|
2013-09-25 19:42:22 +02:00
|
|
|
for (opt= rules; rules && opt->name; opt++)
|
2010-04-08 14:10:05 +02:00
|
|
|
{
|
2013-04-09 23:27:24 +02:00
|
|
|
bool seen=false;
|
|
|
|
for (val= *option_list; val; val= val->next)
|
2010-04-08 14:10:05 +02:00
|
|
|
{
|
2013-04-09 23:27:24 +02:00
|
|
|
last= val;
|
2020-01-26 20:27:13 +04:00
|
|
|
if (system_charset_info->strnncoll(opt->name, opt->name_length,
|
|
|
|
val->name.str, val->name.length))
|
2010-04-08 14:10:05 +02:00
|
|
|
continue;
|
|
|
|
|
2015-04-10 02:36:54 +02:00
|
|
|
/* skip duplicates (see engine_option_value constructor above) */
|
2013-04-09 23:27:24 +02:00
|
|
|
if (val->parsed && !val->value.str)
|
|
|
|
continue;
|
|
|
|
|
2010-04-08 14:10:05 +02:00
|
|
|
if (set_one_value(opt, thd, &val->value,
|
|
|
|
*option_struct, suppress_warning || val->parsed, root))
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
val->parsed= true;
|
2015-04-10 02:36:54 +02:00
|
|
|
seen=true;
|
2010-04-08 14:10:05 +02:00
|
|
|
break;
|
|
|
|
}
|
2015-04-10 02:36:54 +02:00
|
|
|
if (!seen || (opt->var && !last->value.str))
|
2024-10-17 15:57:03 +02:00
|
|
|
set_one_value(opt, thd, &null_clex_str, *option_struct,
|
2013-04-09 23:27:24 +02:00
|
|
|
suppress_warning, root);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (val= *option_list; val; val= val->next)
|
|
|
|
{
|
2010-04-08 14:10:05 +02:00
|
|
|
if (report_unknown_option(thd, val, suppress_warning))
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
val->parsed= true;
|
|
|
|
}
|
|
|
|
|
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-09 23:27:24 +02:00
|
|
|
/**
|
|
|
|
Resolves all HA_OPTION_TYPE_SYSVAR elements.
|
|
|
|
|
|
|
|
This is done when an engine is loaded.
|
|
|
|
*/
|
|
|
|
static bool resolve_sysvars(handlerton *hton, ha_create_table_option *rules)
|
|
|
|
{
|
2013-09-25 19:42:22 +02:00
|
|
|
for (ha_create_table_option *opt= rules; rules && opt->name; opt++)
|
2013-04-09 23:27:24 +02:00
|
|
|
{
|
|
|
|
if (opt->type == HA_OPTION_TYPE_SYSVAR)
|
|
|
|
{
|
|
|
|
struct my_option optp;
|
|
|
|
plugin_opt_set_limits(&optp, opt->var);
|
|
|
|
switch(optp.var_type) {
|
|
|
|
case GET_ULL:
|
|
|
|
case GET_ULONG:
|
|
|
|
case GET_UINT:
|
|
|
|
opt->type= HA_OPTION_TYPE_ULL;
|
|
|
|
opt->def_value= (ulonglong)optp.def_value;
|
|
|
|
opt->min_value= (ulonglong)optp.min_value;
|
|
|
|
opt->max_value= (ulonglong)optp.max_value;
|
|
|
|
opt->block_size= (ulonglong)optp.block_size;
|
|
|
|
break;
|
|
|
|
case GET_STR:
|
|
|
|
case GET_STR_ALLOC:
|
|
|
|
opt->type= HA_OPTION_TYPE_STRING;
|
|
|
|
break;
|
|
|
|
case GET_BOOL:
|
|
|
|
opt->type= HA_OPTION_TYPE_BOOL;
|
|
|
|
opt->def_value= optp.def_value;
|
|
|
|
break;
|
|
|
|
case GET_ENUM:
|
|
|
|
{
|
|
|
|
opt->type= HA_OPTION_TYPE_ENUM;
|
|
|
|
opt->def_value= optp.def_value;
|
|
|
|
|
|
|
|
char buf[256];
|
|
|
|
String str(buf, sizeof(buf), system_charset_info);
|
2014-03-26 09:33:03 +01:00
|
|
|
str.length(0);
|
2013-04-09 23:27:24 +02:00
|
|
|
for (const char **s= optp.typelib->type_names; *s; s++)
|
|
|
|
{
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str.append(*s, strlen(*s)) || str.append(','))
|
2013-04-09 23:27:24 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
DBUG_ASSERT(str.length());
|
2020-01-29 13:50:26 +01:00
|
|
|
opt->values= my_strndup(PSI_INSTRUMENT_ME, str.ptr(), str.length()-1, MYF(MY_WME));
|
2013-04-09 23:27:24 +02:00
|
|
|
if (!opt->values)
|
|
|
|
return 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
DBUG_ASSERT(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool resolve_sysvar_table_options(handlerton *hton)
|
|
|
|
{
|
|
|
|
return resolve_sysvars(hton, hton->table_options) ||
|
|
|
|
resolve_sysvars(hton, hton->field_options) ||
|
|
|
|
resolve_sysvars(hton, hton->index_options);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Restore HA_OPTION_TYPE_SYSVAR options back as they were
|
|
|
|
before resolve_sysvars().
|
|
|
|
|
|
|
|
This is done when the engine is unloaded, so that we could
|
|
|
|
call resolve_sysvars() if the engine is installed again.
|
|
|
|
*/
|
|
|
|
static void free_sysvars(handlerton *hton, ha_create_table_option *rules)
|
|
|
|
{
|
2013-09-25 19:42:22 +02:00
|
|
|
for (ha_create_table_option *opt= rules; rules && opt->name; opt++)
|
2013-04-09 23:27:24 +02:00
|
|
|
{
|
|
|
|
if (opt->var)
|
|
|
|
{
|
|
|
|
my_free(const_cast<char*>(opt->values));
|
|
|
|
opt->type= HA_OPTION_TYPE_SYSVAR;
|
|
|
|
opt->def_value= 0;
|
|
|
|
opt->min_value= 0;
|
|
|
|
opt->max_value= 0;
|
|
|
|
opt->block_size= 0;
|
|
|
|
opt->values= 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void free_sysvar_table_options(handlerton *hton)
|
|
|
|
{
|
|
|
|
free_sysvars(hton, hton->table_options);
|
|
|
|
free_sysvars(hton, hton->field_options);
|
|
|
|
free_sysvars(hton, hton->index_options);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-04-08 14:10:05 +02:00
|
|
|
/**
|
|
|
|
Parses all table/fields/keys options
|
|
|
|
|
|
|
|
@param thd thread handler
|
|
|
|
@param file handler of the table
|
|
|
|
@parem share descriptor of the table
|
|
|
|
|
|
|
|
@retval TRUE Error
|
|
|
|
@retval FALSE OK
|
|
|
|
*/
|
|
|
|
|
2013-04-09 23:27:14 +02:00
|
|
|
bool parse_engine_table_options(THD *thd, handlerton *ht, TABLE_SHARE *share)
|
2010-04-08 14:10:05 +02:00
|
|
|
{
|
|
|
|
MEM_ROOT *root= &share->mem_root;
|
|
|
|
DBUG_ENTER("parse_engine_table_options");
|
|
|
|
|
2024-10-17 15:57:03 +02:00
|
|
|
if (parse_option_list(thd, &share->option_struct, & share->option_list,
|
2010-04-08 14:10:05 +02:00
|
|
|
ht->table_options, TRUE, root))
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
|
|
|
|
for (Field **field= share->field; *field; field++)
|
|
|
|
{
|
2024-10-17 15:57:03 +02:00
|
|
|
if (parse_option_list(thd, &(*field)->option_struct,
|
2013-04-09 23:27:24 +02:00
|
|
|
& (*field)->option_list,
|
2010-04-08 14:10:05 +02:00
|
|
|
ht->field_options, TRUE, root))
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint index= 0; index < share->keys; index ++)
|
|
|
|
{
|
2024-10-17 15:57:03 +02:00
|
|
|
if (parse_option_list(thd, &share->key_info[index].option_struct,
|
2013-04-09 23:27:24 +02:00
|
|
|
& share->key_info[index].option_list,
|
2010-04-08 14:10:05 +02:00
|
|
|
ht->index_options, TRUE, root))
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
DBUG_RETURN(FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-25 19:42:22 +02:00
|
|
|
bool engine_options_differ(void *old_struct, void *new_struct,
|
|
|
|
ha_create_table_option *rules)
|
|
|
|
{
|
|
|
|
ha_create_table_option *opt;
|
|
|
|
for (opt= rules; rules && opt->name; opt++)
|
|
|
|
{
|
|
|
|
char **old_val= (char**)value_ptr(old_struct, opt);
|
|
|
|
char **new_val= (char**)value_ptr(new_struct, opt);
|
|
|
|
int neq;
|
|
|
|
if (opt->type == HA_OPTION_TYPE_STRING)
|
|
|
|
neq= (*old_val && *new_val) ? strcmp(*old_val, *new_val) : *old_val != *new_val;
|
|
|
|
else
|
|
|
|
neq= memcmp(old_val, new_val, ha_option_type_sizeof[opt->type]);
|
|
|
|
if (neq)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-04-08 14:10:05 +02:00
|
|
|
/**
|
|
|
|
Returns representation length of key and value in the frm file
|
|
|
|
*/
|
|
|
|
|
|
|
|
uint engine_option_value::frm_length()
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
1 byte - name length
|
|
|
|
2 bytes - value length
|
|
|
|
|
|
|
|
if value.str is NULL, this option is not written to frm (=DEFAULT)
|
|
|
|
*/
|
2018-02-06 12:55:58 +00:00
|
|
|
return value.str ? (uint)(1 + name.length + 2 + value.length) : 0;
|
2010-04-08 14:10:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Returns length of representation of option list in the frm file
|
|
|
|
*/
|
|
|
|
|
|
|
|
static uint option_list_frm_length(engine_option_value *opt)
|
|
|
|
{
|
|
|
|
uint res= 0;
|
|
|
|
|
|
|
|
for (; opt; opt= opt->next)
|
|
|
|
res+= opt->frm_length();
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Calculates length of options image in the .frm
|
|
|
|
|
|
|
|
@param table_option_list list of table options
|
|
|
|
@param create_fields field descriptors list
|
|
|
|
@param keys number of keys
|
|
|
|
@param key_info array of key descriptors
|
|
|
|
|
|
|
|
@returns length of image in frm
|
|
|
|
*/
|
|
|
|
|
|
|
|
uint engine_table_options_frm_length(engine_option_value *table_option_list,
|
|
|
|
List<Create_field> &create_fields,
|
|
|
|
uint keys, KEY *key_info)
|
|
|
|
{
|
|
|
|
List_iterator<Create_field> it(create_fields);
|
|
|
|
Create_field *field;
|
|
|
|
uint res, index;
|
|
|
|
DBUG_ENTER("engine_table_options_frm_length");
|
|
|
|
|
|
|
|
res= option_list_frm_length(table_option_list);
|
|
|
|
|
|
|
|
while ((field= it++))
|
|
|
|
res+= option_list_frm_length(field->option_list);
|
|
|
|
|
|
|
|
for (index= 0; index < keys; index++, key_info++)
|
|
|
|
res+= option_list_frm_length(key_info->option_list);
|
|
|
|
|
|
|
|
/*
|
|
|
|
if there's at least one option somewhere (res > 0)
|
|
|
|
we write option lists for all fields and keys, zero-terminated.
|
|
|
|
If there're no options we write nothing at all (backward compatibility)
|
|
|
|
*/
|
|
|
|
DBUG_RETURN(res ? res + 1 + create_fields.elements + keys : 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Writes image of the key and value to the frm image buffer
|
|
|
|
|
|
|
|
@param buff pointer to the buffer free space beginning
|
|
|
|
|
|
|
|
@returns pointer to byte after last recorded in the buffer
|
|
|
|
*/
|
|
|
|
|
|
|
|
uchar *engine_option_value::frm_image(uchar *buff)
|
|
|
|
{
|
|
|
|
if (value.str)
|
|
|
|
{
|
2018-01-15 18:59:27 +00:00
|
|
|
DBUG_ASSERT(name.length <= 0xff);
|
|
|
|
*buff++= (uchar)name.length;
|
2010-04-08 14:10:05 +02:00
|
|
|
memcpy(buff, name.str, name.length);
|
|
|
|
buff+= name.length;
|
|
|
|
int2store(buff, value.length | (quoted_value ? FRM_QUOTED_VALUE : 0));
|
|
|
|
buff+= 2;
|
|
|
|
memcpy(buff, (const uchar *) value.str, value.length);
|
|
|
|
buff+= value.length;
|
|
|
|
}
|
|
|
|
return buff;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Writes image of the key and value to the frm image buffer
|
|
|
|
|
|
|
|
@param buff pointer to the buffer to store the options in
|
|
|
|
@param opt list of options;
|
|
|
|
|
|
|
|
@returns pointer to the end of the stored data in the buffer
|
|
|
|
*/
|
|
|
|
static uchar *option_list_frm_image(uchar *buff, engine_option_value *opt)
|
|
|
|
{
|
|
|
|
for (; opt; opt= opt->next)
|
|
|
|
buff= opt->frm_image(buff);
|
|
|
|
|
|
|
|
*buff++= 0;
|
|
|
|
return buff;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Writes options image in the .frm buffer
|
|
|
|
|
|
|
|
@param buff pointer to the buffer
|
|
|
|
@param table_option_list list of table options
|
|
|
|
@param create_fields field descriptors list
|
|
|
|
@param keys number of keys
|
|
|
|
@param key_info array of key descriptors
|
|
|
|
|
|
|
|
@returns pointer to byte after last recorded in the buffer
|
|
|
|
*/
|
|
|
|
|
|
|
|
uchar *engine_table_options_frm_image(uchar *buff,
|
|
|
|
engine_option_value *table_option_list,
|
|
|
|
List<Create_field> &create_fields,
|
|
|
|
uint keys, KEY *key_info)
|
|
|
|
{
|
|
|
|
List_iterator<Create_field> it(create_fields);
|
|
|
|
Create_field *field;
|
|
|
|
KEY *key_info_end= key_info + keys;
|
|
|
|
DBUG_ENTER("engine_table_options_frm_image");
|
|
|
|
|
|
|
|
buff= option_list_frm_image(buff, table_option_list);
|
|
|
|
|
|
|
|
while ((field= it++))
|
|
|
|
buff= option_list_frm_image(buff, field->option_list);
|
|
|
|
|
|
|
|
while (key_info < key_info_end)
|
|
|
|
buff= option_list_frm_image(buff, (key_info++)->option_list);
|
|
|
|
|
|
|
|
DBUG_RETURN(buff);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Reads name and value from buffer, then link it in the list
|
|
|
|
|
|
|
|
@param buff the buffer to read from
|
|
|
|
@param start The list beginning or NULL
|
|
|
|
@param end The list last element or does not matter
|
|
|
|
@param root MEM_ROOT for allocating
|
|
|
|
|
|
|
|
@returns pointer to byte after last recorded in the buffer
|
|
|
|
*/
|
2016-05-04 16:05:30 +02:00
|
|
|
uchar *engine_option_value::frm_read(const uchar *buff, const uchar *buff_end,
|
|
|
|
engine_option_value **start,
|
2010-04-08 14:10:05 +02:00
|
|
|
engine_option_value **end, MEM_ROOT *root)
|
|
|
|
{
|
2017-04-23 19:39:57 +03:00
|
|
|
LEX_CSTRING name, value;
|
2010-04-08 14:10:05 +02:00
|
|
|
uint len;
|
2016-05-04 16:05:30 +02:00
|
|
|
#define need_buff(N) if (buff + (N) >= buff_end) return NULL
|
2010-04-08 14:10:05 +02:00
|
|
|
|
2016-05-04 16:05:30 +02:00
|
|
|
need_buff(3);
|
2010-04-08 14:10:05 +02:00
|
|
|
name.length= buff[0];
|
|
|
|
buff++;
|
2016-05-04 16:05:30 +02:00
|
|
|
need_buff(name.length + 2);
|
2010-04-08 14:10:05 +02:00
|
|
|
if (!(name.str= strmake_root(root, (const char*)buff, name.length)))
|
|
|
|
return NULL;
|
|
|
|
buff+= name.length;
|
|
|
|
len= uint2korr(buff);
|
|
|
|
value.length= len & ~FRM_QUOTED_VALUE;
|
|
|
|
buff+= 2;
|
2016-05-04 16:05:30 +02:00
|
|
|
need_buff(value.length);
|
2010-04-08 14:10:05 +02:00
|
|
|
if (!(value.str= strmake_root(root, (const char*)buff, value.length)))
|
|
|
|
return NULL;
|
|
|
|
buff+= value.length;
|
|
|
|
|
|
|
|
engine_option_value *ptr=new (root)
|
|
|
|
engine_option_value(name, value, len & FRM_QUOTED_VALUE, start, end);
|
|
|
|
if (!ptr)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return (uchar *)buff;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Reads options from this buffer
|
|
|
|
|
|
|
|
@param buff the buffer to read from
|
|
|
|
@param length buffer length
|
|
|
|
@param share table descriptor
|
|
|
|
@param root MEM_ROOT for allocating
|
|
|
|
|
|
|
|
@retval TRUE Error
|
|
|
|
@retval FALSE OK
|
|
|
|
*/
|
|
|
|
|
2018-02-06 12:55:58 +00:00
|
|
|
bool engine_table_options_frm_read(const uchar *buff, size_t length,
|
2013-04-09 23:27:14 +02:00
|
|
|
TABLE_SHARE *share)
|
2010-04-08 14:10:05 +02:00
|
|
|
{
|
|
|
|
const uchar *buff_end= buff + length;
|
2011-11-27 17:50:08 +01:00
|
|
|
engine_option_value *UNINIT_VAR(end);
|
2010-04-08 14:10:05 +02:00
|
|
|
MEM_ROOT *root= &share->mem_root;
|
|
|
|
uint count;
|
|
|
|
DBUG_ENTER("engine_table_options_frm_read");
|
|
|
|
|
|
|
|
while (buff < buff_end && *buff)
|
|
|
|
{
|
2016-05-04 16:05:30 +02:00
|
|
|
if (!(buff= engine_option_value::frm_read(buff, buff_end,
|
|
|
|
&share->option_list, &end, root)))
|
2010-04-08 14:10:05 +02:00
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
|
|
|
buff++;
|
|
|
|
|
|
|
|
for (count=0; count < share->fields; count++)
|
|
|
|
{
|
|
|
|
while (buff < buff_end && *buff)
|
|
|
|
{
|
2016-05-04 16:05:30 +02:00
|
|
|
if (!(buff= engine_option_value::frm_read(buff, buff_end,
|
2010-04-08 14:10:05 +02:00
|
|
|
&share->field[count]->option_list,
|
|
|
|
&end, root)))
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
|
|
|
buff++;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (count=0; count < share->keys; count++)
|
|
|
|
{
|
|
|
|
while (buff < buff_end && *buff)
|
|
|
|
{
|
2016-05-04 16:05:30 +02:00
|
|
|
if (!(buff= engine_option_value::frm_read(buff, buff_end,
|
2010-04-08 14:10:05 +02:00
|
|
|
&share->key_info[count].option_list,
|
|
|
|
&end, root)))
|
|
|
|
DBUG_RETURN(TRUE);
|
|
|
|
}
|
|
|
|
buff++;
|
|
|
|
}
|
|
|
|
|
2010-05-12 18:15:35 +02:00
|
|
|
if (buff < buff_end)
|
2010-07-17 01:41:44 +03:00
|
|
|
sql_print_warning("Table '%s' was created in a later MariaDB version - "
|
2010-05-12 18:15:35 +02:00
|
|
|
"unknown table attributes were ignored",
|
2010-07-17 01:41:44 +03:00
|
|
|
share->table_name.str);
|
2010-05-12 18:15:35 +02:00
|
|
|
|
|
|
|
DBUG_RETURN(buff > buff_end);
|
2010-04-08 14:10:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Merges two lists of engine_option_value's with duplicate removal.
|
|
|
|
*/
|
|
|
|
|
|
|
|
engine_option_value *merge_engine_table_options(engine_option_value *first,
|
|
|
|
engine_option_value *second,
|
|
|
|
MEM_ROOT *root)
|
|
|
|
{
|
2015-02-10 14:05:49 +04:00
|
|
|
engine_option_value *UNINIT_VAR(end), *opt;
|
2010-04-08 14:10:05 +02:00
|
|
|
DBUG_ENTER("merge_engine_table_options");
|
|
|
|
|
2013-07-26 23:02:48 +04:00
|
|
|
/* Create copy of first list */
|
|
|
|
for (opt= first, first= 0; opt; opt= opt->next)
|
|
|
|
new (root) engine_option_value(opt, &first, &end);
|
2010-04-08 14:10:05 +02:00
|
|
|
|
|
|
|
for (opt= second; opt; opt= opt->next)
|
|
|
|
new (root) engine_option_value(opt->name, opt->value, opt->quoted_value,
|
|
|
|
&first, &end);
|
|
|
|
DBUG_RETURN(first);
|
|
|
|
}
|
2014-07-08 19:39:27 +02:00
|
|
|
|
|
|
|
bool is_engine_option_known(engine_option_value *opt,
|
|
|
|
ha_create_table_option *rules)
|
|
|
|
{
|
|
|
|
if (!rules)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (; rules->name; rules++)
|
|
|
|
{
|
2020-01-26 20:27:13 +04:00
|
|
|
if (!system_charset_info->strnncoll(rules->name, rules->name_length,
|
|
|
|
opt->name.str, opt->name.length))
|
2014-07-08 19:39:27 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|