mirror of
https://github.com/MariaDB/server.git
synced 2025-01-23 07:14:17 +01:00
7b96416f3c
This fixed several cases where we where using just ulong for sql_mode
427 lines
14 KiB
C++
427 lines
14 KiB
C++
#ifndef SET_VAR_INCLUDED
|
|
#define SET_VAR_INCLUDED
|
|
/* Copyright (c) 2002, 2013, Oracle and/or its affiliates.
|
|
Copyright (c) 2009, 2014, SkySQL Ab.
|
|
|
|
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 Street, Fifth Floor, Boston, MA 02110-1301, USA */
|
|
|
|
/**
|
|
@file
|
|
"public" interface to sys_var - server configuration variables.
|
|
*/
|
|
|
|
#ifdef USE_PRAGMA_INTERFACE
|
|
#pragma interface /* gcc class implementation */
|
|
#endif
|
|
|
|
#include <my_getopt.h>
|
|
|
|
class sys_var;
|
|
class set_var;
|
|
class sys_var_pluginvar;
|
|
class PolyLock;
|
|
class Item_func_set_user_var;
|
|
|
|
// This include needs to be here since item.h requires enum_var_type :-P
|
|
#include "item.h" /* Item */
|
|
#include "sql_class.h" /* THD */
|
|
|
|
extern TYPELIB bool_typelib;
|
|
|
|
struct sys_var_chain
|
|
{
|
|
sys_var *first;
|
|
sys_var *last;
|
|
};
|
|
|
|
int mysql_add_sys_var_chain(sys_var *chain);
|
|
int mysql_del_sys_var_chain(sys_var *chain);
|
|
|
|
|
|
/**
|
|
A class representing one system variable - that is something
|
|
that can be accessed as @@global.variable_name or @@session.variable_name,
|
|
visible in SHOW xxx VARIABLES and in INFORMATION_SCHEMA.xxx_VARIABLES,
|
|
optionally it can be assigned to, optionally it can have a command-line
|
|
counterpart with the same name.
|
|
*/
|
|
class sys_var: protected Value_source // for double_from_string_with_check
|
|
{
|
|
public:
|
|
sys_var *next;
|
|
LEX_CSTRING name;
|
|
bool *test_load;
|
|
enum flag_enum { GLOBAL, SESSION, ONLY_SESSION, SCOPE_MASK=1023,
|
|
READONLY=1024, ALLOCATED=2048, PARSE_EARLY=4096,
|
|
NO_SET_STATEMENT=8192, AUTO_SET=16384};
|
|
enum { NO_GETOPT=-1, GETOPT_ONLY_HELP=-2 };
|
|
enum where { CONFIG, AUTO, SQL, COMPILE_TIME, ENV };
|
|
|
|
/**
|
|
Enumeration type to indicate for a system variable whether
|
|
it will be written to the binlog or not.
|
|
*/
|
|
enum binlog_status_enum { VARIABLE_NOT_IN_BINLOG,
|
|
SESSION_VARIABLE_IN_BINLOG } binlog_status;
|
|
|
|
my_option option; ///< min, max, default values are stored here
|
|
enum where value_origin;
|
|
|
|
protected:
|
|
typedef bool (*on_check_function)(sys_var *self, THD *thd, set_var *var);
|
|
typedef bool (*on_update_function)(sys_var *self, THD *thd, enum_var_type type);
|
|
|
|
int flags; ///< or'ed flag_enum values
|
|
const SHOW_TYPE show_val_type; ///< what value_ptr() returns for sql_show.cc
|
|
PolyLock *guard; ///< *second* lock that protects the variable
|
|
ptrdiff_t offset; ///< offset to the value from global_system_variables
|
|
on_check_function on_check;
|
|
on_update_function on_update;
|
|
const char *const deprecation_substitute;
|
|
bool is_os_charset; ///< true if the value is in character_set_filesystem
|
|
|
|
public:
|
|
sys_var(sys_var_chain *chain, const char *name_arg, const char *comment,
|
|
int flag_args, ptrdiff_t off, int getopt_id,
|
|
enum get_opt_arg_type getopt_arg_type, SHOW_TYPE show_val_type_arg,
|
|
longlong def_val, PolyLock *lock, enum binlog_status_enum binlog_status_arg,
|
|
on_check_function on_check_func, on_update_function on_update_func,
|
|
const char *substitute);
|
|
|
|
virtual ~sys_var() {}
|
|
|
|
/**
|
|
All the cleanup procedures should be performed here
|
|
*/
|
|
virtual void cleanup() {}
|
|
/**
|
|
downcast for sys_var_pluginvar. Returns this if it's an instance
|
|
of sys_var_pluginvar, and 0 otherwise.
|
|
*/
|
|
virtual sys_var_pluginvar *cast_pluginvar() { return 0; }
|
|
|
|
bool check(THD *thd, set_var *var);
|
|
uchar *value_ptr(THD *thd, enum_var_type type, const LEX_STRING *base);
|
|
|
|
/**
|
|
Update the system variable with the default value from either
|
|
session or global scope. The default value is stored in the
|
|
'var' argument. Return false when successful.
|
|
*/
|
|
bool set_default(THD *thd, set_var *var);
|
|
bool update(THD *thd, set_var *var);
|
|
|
|
String *val_str_nolock(String *str, THD *thd, const uchar *value);
|
|
longlong val_int(bool *is_null, THD *thd, enum_var_type type, const LEX_STRING *base);
|
|
String *val_str(String *str, THD *thd, enum_var_type type, const LEX_STRING *base);
|
|
double val_real(bool *is_null, THD *thd, enum_var_type type, const LEX_STRING *base);
|
|
|
|
SHOW_TYPE show_type() { return show_val_type; }
|
|
int scope() const { return flags & SCOPE_MASK; }
|
|
CHARSET_INFO *charset(THD *thd);
|
|
bool is_readonly() const { return flags & READONLY; }
|
|
/**
|
|
the following is only true for keycache variables,
|
|
that support the syntax @@keycache_name.variable_name
|
|
*/
|
|
bool is_struct() { return option.var_type & GET_ASK_ADDR; }
|
|
bool is_set_stmt_ok() const { return !(flags & NO_SET_STATEMENT); }
|
|
bool is_written_to_binlog(enum_var_type type)
|
|
{ return type != OPT_GLOBAL && binlog_status == SESSION_VARIABLE_IN_BINLOG; }
|
|
bool check_update_type(const Item *item)
|
|
{
|
|
Item_result type= item->result_type();
|
|
switch (option.var_type & GET_TYPE_MASK) {
|
|
case GET_INT:
|
|
case GET_UINT:
|
|
case GET_LONG:
|
|
case GET_ULONG:
|
|
case GET_LL:
|
|
case GET_ULL:
|
|
return type != INT_RESULT &&
|
|
(type != DECIMAL_RESULT || item->decimals != 0);
|
|
case GET_STR:
|
|
case GET_STR_ALLOC:
|
|
return type != STRING_RESULT;
|
|
case GET_ENUM:
|
|
case GET_BOOL:
|
|
case GET_SET:
|
|
case GET_FLAGSET:
|
|
return type != STRING_RESULT && type != INT_RESULT;
|
|
case GET_DOUBLE:
|
|
return type != INT_RESULT && type != REAL_RESULT && type != DECIMAL_RESULT;
|
|
default:
|
|
return true;
|
|
}
|
|
}
|
|
|
|
bool check_type(enum_var_type type)
|
|
{
|
|
switch (scope())
|
|
{
|
|
case GLOBAL: return type != OPT_GLOBAL;
|
|
case SESSION: return false; // always ok
|
|
case ONLY_SESSION: return type == OPT_GLOBAL;
|
|
}
|
|
return true; // keep gcc happy
|
|
}
|
|
bool register_option(DYNAMIC_ARRAY *array, int parse_flags)
|
|
{
|
|
DBUG_ASSERT(parse_flags == GETOPT_ONLY_HELP ||
|
|
parse_flags == PARSE_EARLY || parse_flags == 0);
|
|
if (option.id == NO_GETOPT)
|
|
return 0;
|
|
if (parse_flags == GETOPT_ONLY_HELP)
|
|
{
|
|
if (option.id != GETOPT_ONLY_HELP)
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
if (option.id == GETOPT_ONLY_HELP)
|
|
return 0;
|
|
if ((flags & PARSE_EARLY) != parse_flags)
|
|
return 0;
|
|
}
|
|
return insert_dynamic(array, (uchar*)&option);
|
|
}
|
|
void do_deprecated_warning(THD *thd);
|
|
/**
|
|
whether session value of a sysvar is a default one.
|
|
|
|
in this simple implementation we don't distinguish between default
|
|
and non-default values. for most variables it's ok, they don't treat
|
|
default values specially. this method is overwritten in descendant
|
|
classes as necessary.
|
|
*/
|
|
virtual bool session_is_default(THD *thd) { return false; }
|
|
|
|
virtual uchar *default_value_ptr(THD *thd)
|
|
{ return (uchar*)&option.def_value; }
|
|
|
|
private:
|
|
virtual bool do_check(THD *thd, set_var *var) = 0;
|
|
/**
|
|
save the session default value of the variable in var
|
|
*/
|
|
virtual void session_save_default(THD *thd, set_var *var) = 0;
|
|
/**
|
|
save the global default value of the variable in var
|
|
*/
|
|
virtual void global_save_default(THD *thd, set_var *var) = 0;
|
|
virtual bool session_update(THD *thd, set_var *var) = 0;
|
|
virtual bool global_update(THD *thd, set_var *var) = 0;
|
|
|
|
protected:
|
|
/**
|
|
A pointer to a value of the variable for SHOW.
|
|
It must be of show_val_type type (my_bool for SHOW_MY_BOOL,
|
|
int for SHOW_INT, longlong for SHOW_LONGLONG, etc).
|
|
*/
|
|
virtual uchar *session_value_ptr(THD *thd, const LEX_STRING *base);
|
|
virtual uchar *global_value_ptr(THD *thd, const LEX_STRING *base);
|
|
|
|
/**
|
|
A pointer to a storage area of the variable, to the raw data.
|
|
Typically it's the same as session_value_ptr(), but it's different,
|
|
for example, for ENUM, that is printed as a string, but stored as a number.
|
|
*/
|
|
uchar *session_var_ptr(THD *thd)
|
|
{ return ((uchar*)&(thd->variables)) + offset; }
|
|
|
|
uchar *global_var_ptr()
|
|
{ return ((uchar*)&global_system_variables) + offset; }
|
|
|
|
friend class Session_sysvars_tracker;
|
|
friend class Session_tracker;
|
|
};
|
|
|
|
#include "sql_plugin.h" /* SHOW_HA_ROWS, SHOW_MY_BOOL */
|
|
|
|
|
|
/****************************************************************************
|
|
Classes for parsing of the SET command
|
|
****************************************************************************/
|
|
|
|
/**
|
|
A base class for everything that can be set with SET command.
|
|
It's similar to Items, an instance of this is created by the parser
|
|
for every assigmnent in SET (or elsewhere, e.g. in SELECT).
|
|
*/
|
|
class set_var_base :public Sql_alloc
|
|
{
|
|
public:
|
|
set_var_base() {}
|
|
virtual ~set_var_base() {}
|
|
virtual int check(THD *thd)=0; /* To check privileges etc. */
|
|
virtual int update(THD *thd)=0; /* To set the value */
|
|
virtual int light_check(THD *thd) { return check(thd); } /* for PS */
|
|
virtual bool is_system() { return FALSE; }
|
|
};
|
|
|
|
|
|
/**
|
|
set_var_base descendant for assignments to the system variables.
|
|
*/
|
|
class set_var :public set_var_base
|
|
{
|
|
public:
|
|
sys_var *var; ///< system variable to be updated
|
|
Item *value; ///< the expression that provides the new value of the variable
|
|
enum_var_type type;
|
|
union ///< temp storage to hold a value between sys_var::check and ::update
|
|
{
|
|
ulonglong ulonglong_value; ///< for unsigned integer, set, enum sysvars
|
|
longlong longlong_value; ///< for signed integer
|
|
double double_value; ///< for Sys_var_double
|
|
plugin_ref plugin; ///< for Sys_var_plugin
|
|
Time_zone *time_zone; ///< for Sys_var_tz
|
|
LEX_STRING string_value; ///< for Sys_var_charptr and others
|
|
const void *ptr; ///< for Sys_var_struct
|
|
} save_result;
|
|
LEX_STRING base; /**< for structured variables, like keycache_name.variable_name */
|
|
|
|
set_var(THD *thd, enum_var_type type_arg, sys_var *var_arg,
|
|
const LEX_STRING *base_name_arg, Item *value_arg);
|
|
virtual bool is_system() { return 1; }
|
|
int check(THD *thd);
|
|
int update(THD *thd);
|
|
int light_check(THD *thd);
|
|
};
|
|
|
|
|
|
/* User variables like @my_own_variable */
|
|
class set_var_user: public set_var_base
|
|
{
|
|
Item_func_set_user_var *user_var_item;
|
|
public:
|
|
set_var_user(Item_func_set_user_var *item)
|
|
:user_var_item(item)
|
|
{}
|
|
int check(THD *thd);
|
|
int update(THD *thd);
|
|
int light_check(THD *thd);
|
|
};
|
|
|
|
/* For SET PASSWORD */
|
|
|
|
class set_var_password: public set_var_base
|
|
{
|
|
LEX_USER *user;
|
|
public:
|
|
set_var_password(LEX_USER *user_arg) :user(user_arg)
|
|
{}
|
|
int check(THD *thd);
|
|
int update(THD *thd);
|
|
};
|
|
|
|
/* For SET ROLE */
|
|
|
|
class set_var_role: public set_var_base
|
|
{
|
|
LEX_STRING role;
|
|
ulonglong access;
|
|
public:
|
|
set_var_role(LEX_STRING role_arg) : role(role_arg) {}
|
|
int check(THD *thd);
|
|
int update(THD *thd);
|
|
};
|
|
|
|
/* For SET DEFAULT ROLE */
|
|
|
|
class set_var_default_role: public set_var_base
|
|
{
|
|
LEX_USER *user, *real_user;
|
|
LEX_STRING role;
|
|
public:
|
|
set_var_default_role(LEX_USER *user_arg, LEX_STRING role_arg) :
|
|
user(user_arg), role(role_arg) {}
|
|
int check(THD *thd);
|
|
int update(THD *thd);
|
|
};
|
|
|
|
/* For SET NAMES and SET CHARACTER SET */
|
|
|
|
class set_var_collation_client: public set_var_base
|
|
{
|
|
CHARSET_INFO *character_set_client;
|
|
CHARSET_INFO *character_set_results;
|
|
CHARSET_INFO *collation_connection;
|
|
public:
|
|
set_var_collation_client(CHARSET_INFO *client_coll_arg,
|
|
CHARSET_INFO *connection_coll_arg,
|
|
CHARSET_INFO *result_coll_arg)
|
|
:character_set_client(client_coll_arg),
|
|
character_set_results(result_coll_arg),
|
|
collation_connection(connection_coll_arg)
|
|
{}
|
|
int check(THD *thd);
|
|
int update(THD *thd);
|
|
};
|
|
|
|
|
|
/* optional things, have_* variables */
|
|
extern SHOW_COMP_OPTION have_csv, have_innodb;
|
|
extern SHOW_COMP_OPTION have_ndbcluster, have_partitioning;
|
|
extern SHOW_COMP_OPTION have_profiling;
|
|
|
|
extern SHOW_COMP_OPTION have_ssl, have_symlink, have_dlopen;
|
|
extern SHOW_COMP_OPTION have_query_cache;
|
|
extern SHOW_COMP_OPTION have_geometry, have_rtree_keys;
|
|
extern SHOW_COMP_OPTION have_crypt;
|
|
extern SHOW_COMP_OPTION have_compress;
|
|
extern SHOW_COMP_OPTION have_openssl;
|
|
|
|
/*
|
|
Prototypes for helper functions
|
|
*/
|
|
|
|
SHOW_VAR* enumerate_sys_vars(THD *thd, bool sorted, enum enum_var_type type);
|
|
int fill_sysvars(THD *thd, TABLE_LIST *tables, COND *cond);
|
|
|
|
sys_var *find_sys_var(THD *thd, const char *str, size_t length=0);
|
|
int sql_set_variables(THD *thd, List<set_var_base> *var_list, bool free);
|
|
|
|
#define SYSVAR_AUTOSIZE(VAR,VAL) \
|
|
do { \
|
|
VAR= (VAL); \
|
|
set_sys_var_value_origin(&VAR, sys_var::AUTO); \
|
|
} while(0)
|
|
|
|
void set_sys_var_value_origin(void *ptr, enum sys_var::where here);
|
|
|
|
enum sys_var::where get_sys_var_value_origin(void *ptr);
|
|
inline bool IS_SYSVAR_AUTOSIZE(void *ptr)
|
|
{
|
|
enum sys_var::where res= get_sys_var_value_origin(ptr);
|
|
return (res == sys_var::AUTO || res == sys_var::COMPILE_TIME);
|
|
}
|
|
|
|
bool fix_delay_key_write(sys_var *self, THD *thd, enum_var_type type);
|
|
|
|
sql_mode_t expand_sql_mode(sql_mode_t sql_mode);
|
|
bool sql_mode_string_representation(THD *thd, sql_mode_t sql_mode, LEX_STRING *ls);
|
|
int default_regex_flags_pcre(const THD *thd);
|
|
|
|
extern sys_var *Sys_autocommit_ptr;
|
|
|
|
CHARSET_INFO *get_old_charset_by_name(const char *old_name);
|
|
|
|
int sys_var_init();
|
|
uint sys_var_elements();
|
|
int sys_var_add_options(DYNAMIC_ARRAY *long_options, int parse_flags);
|
|
void sys_var_end(void);
|
|
|
|
#endif
|
|
|