112
stored in bookmark_hash, this structure is never removed from the
113
hash and is used to mark a single offset for a session local variable
114
even if plugins have been uninstalled and reinstalled, repeatedly.
115
This structure is allocated from plugin_mem_root.
117
The key format is as follows:
118
1 byte - variable type code
119
name_len bytes - variable name
122
126
uint32_t version;
126
typedef boost::unordered_map<string, Bookmark> bookmark_unordered_map;
127
static bookmark_unordered_map bookmark_hash;
132
sys_var class for access to all plugin variables visible to the user
134
class sys_var_pluginvar: public sys_var
137
module::Module *plugin;
138
drizzle_sys_var *plugin_var;
140
sys_var_pluginvar(const std::string name_arg,
141
drizzle_sys_var *plugin_var_arg)
142
:sys_var(name_arg), plugin_var(plugin_var_arg) {}
143
sys_var_pluginvar *cast_pluginvar() { return this; }
144
bool is_readonly() const { return plugin_var->flags & PLUGIN_VAR_READONLY; }
145
bool check_type(sql_var_t type)
146
{ return !(plugin_var->flags & PLUGIN_VAR_SessionLOCAL) && type != OPT_GLOBAL; }
147
bool check_update_type(Item_result type);
148
SHOW_TYPE show_type();
149
unsigned char* real_value_ptr(Session *session, sql_var_t type);
150
TYPELIB* plugin_var_typelib(void);
151
unsigned char* value_ptr(Session *session, sql_var_t type,
152
const LEX_STRING *base);
153
bool check(Session *session, set_var *var);
154
bool check_default(sql_var_t)
155
{ return is_readonly(); }
156
void set_default(Session *session, sql_var_t);
157
bool update(Session *session, set_var *var);
132
162
static void plugin_prune_list(vector<string> &plugin_list,
133
163
const vector<string> &plugins_to_remove);
134
164
static bool plugin_load_list(module::Registry ®istry,
135
memory::Root *tmp_root,
165
memory::Root *tmp_root, int *argc, char **argv,
136
166
const set<string> &plugin_list,
137
167
po::options_description &long_options,
138
168
bool builtin= false);
139
169
static int test_plugin_options(memory::Root *, module::Module *,
140
171
po::options_description &long_options);
141
static void unlock_variables(Session *session, drizzle_system_variables *vars);
142
static void cleanup_variables(drizzle_system_variables *vars);
172
static void unlock_variables(Session *session, struct system_variables *vars);
173
static void cleanup_variables(Session *session, struct system_variables *vars);
174
static void plugin_vars_free_values(sys_var *vars);
176
/* declared in set_var.cc */
177
extern sys_var *intern_find_sys_var(const char *str, uint32_t length, bool no_error);
178
extern bool throw_bounds_warning(Session *session, bool fixed, bool unsignd,
179
const std::string &name, int64_t val);
181
static bool throw_bounds_warning(Session *session, bool fixed, bool unsignd,
182
const char *name, int64_t val)
184
const std::string name_str(name);
185
return throw_bounds_warning(session, fixed, unsignd, name_str, val);
188
/****************************************************************************
189
Value type thunks, allows the C world to play in the C++ world
190
****************************************************************************/
192
static int item_value_type(drizzle_value *value)
194
switch (((st_item_value_holder*)value)->item->result_type()) {
196
return DRIZZLE_VALUE_TYPE_INT;
198
return DRIZZLE_VALUE_TYPE_REAL;
200
return DRIZZLE_VALUE_TYPE_STRING;
204
static const char *item_val_str(drizzle_value *value,
205
char *buffer, int *length)
207
String str(buffer, *length, system_charset_info), *res;
208
if (!(res= ((st_item_value_holder*)value)->item->val_str(&str)))
210
*length= res->length();
211
if (res->c_ptr_quick() == buffer)
215
Lets be nice and create a temporary string since the
218
return current_session->strmake(res->c_ptr_quick(), res->length());
222
static int item_val_int(drizzle_value *value, int64_t *buf)
224
Item *item= ((st_item_value_holder*)value)->item;
225
*buf= item->val_int();
232
static int item_val_real(drizzle_value *value, double *buf)
234
Item *item= ((st_item_value_holder*)value)->item;
235
*buf= item->val_real();
145
242
/****************************************************************************
491
560
unlock_variables(NULL, &global_system_variables);
492
561
unlock_variables(NULL, &max_system_variables);
494
cleanup_variables(&global_system_variables);
495
cleanup_variables(&max_system_variables);
563
cleanup_variables(NULL, &global_system_variables);
564
cleanup_variables(NULL, &max_system_variables);
500
569
/* Dispose of the memory */
571
hash_free(&bookmark_hash);
501
572
plugin_mem_root.free_root(MYF(0));
503
574
global_variables_dynamic_size= 0;
577
/****************************************************************************
578
Internal type declarations for variables support
579
****************************************************************************/
581
#undef DRIZZLE_SYSVAR_NAME
582
#define DRIZZLE_SYSVAR_NAME(name) name
583
#define PLUGIN_VAR_TYPEMASK 0x007f
585
static const uint32_t EXTRA_OPTIONS= 1; /* handle the NULL option */
587
typedef DECLARE_DRIZZLE_SYSVAR_BASIC(sysvar_bool_t, bool);
588
typedef DECLARE_DRIZZLE_SessionVAR_BASIC(sessionvar_bool_t, bool);
589
typedef DECLARE_DRIZZLE_SYSVAR_BASIC(sysvar_str_t, char *);
590
typedef DECLARE_DRIZZLE_SessionVAR_BASIC(sessionvar_str_t, char *);
592
typedef DECLARE_DRIZZLE_SessionVAR_TYPELIB(sessionvar_enum_t, unsigned long);
593
typedef DECLARE_DRIZZLE_SessionVAR_TYPELIB(sessionvar_set_t, uint64_t);
595
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_int_t, int);
596
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_long_t, long);
597
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_int64_t_t, int64_t);
598
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_uint_t, uint);
599
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_ulong_t, ulong);
600
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_uint64_t_t, uint64_t);
602
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_int_t, int);
603
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_long_t, long);
604
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_int64_t_t, int64_t);
605
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_uint_t, uint);
606
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_ulong_t, ulong);
607
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_uint64_t_t, uint64_t);
609
typedef bool *(*mysql_sys_var_ptr_p)(Session* a_session, int offset);
612
/****************************************************************************
613
default variable data check and update functions
614
****************************************************************************/
616
static int check_func_bool(Session *, drizzle_sys_var *var,
617
void *save, drizzle_value *value)
619
char buff[STRING_BUFFER_USUAL_SIZE];
620
const char *strvalue= "NULL", *str;
624
if (value->value_type(value) == DRIZZLE_VALUE_TYPE_STRING)
626
length= sizeof(buff);
627
if (!(str= value->val_str(value, buff, &length)) ||
628
(result= find_type(&bool_typelib, str, length, 1)-1) < 0)
637
if (value->val_int(value, &tmp) < 0)
641
internal::llstr(tmp, buff);
647
*(int*)save= -result;
650
my_error(ER_WRONG_VALUE_FOR_VAR, MYF(0), var->name, strvalue);
655
static int check_func_int(Session *session, drizzle_sys_var *var,
656
void *save, drizzle_value *value)
660
struct option options;
661
value->val_int(value, &tmp);
662
plugin_opt_set_limits(&options, var);
664
if (var->flags & PLUGIN_VAR_UNSIGNED)
665
*(uint32_t *)save= (uint32_t) getopt_ull_limit_value((uint64_t) tmp, &options,
668
*(int *)save= (int) getopt_ll_limit_value(tmp, &options, &fixed);
670
return throw_bounds_warning(session, fixed, var->flags & PLUGIN_VAR_UNSIGNED,
671
var->name, (int64_t) tmp);
675
static int check_func_long(Session *session, drizzle_sys_var *var,
676
void *save, drizzle_value *value)
680
struct option options;
681
value->val_int(value, &tmp);
682
plugin_opt_set_limits(&options, var);
684
if (var->flags & PLUGIN_VAR_UNSIGNED)
685
*(ulong *)save= (ulong) getopt_ull_limit_value((uint64_t) tmp, &options,
688
*(long *)save= (long) getopt_ll_limit_value(tmp, &options, &fixed);
690
return throw_bounds_warning(session, fixed, var->flags & PLUGIN_VAR_UNSIGNED,
691
var->name, (int64_t) tmp);
695
static int check_func_int64_t(Session *session, drizzle_sys_var *var,
696
void *save, drizzle_value *value)
700
struct option options;
701
value->val_int(value, &tmp);
702
plugin_opt_set_limits(&options, var);
704
if (var->flags & PLUGIN_VAR_UNSIGNED)
705
*(uint64_t *)save= getopt_ull_limit_value((uint64_t) tmp, &options,
708
*(int64_t *)save= getopt_ll_limit_value(tmp, &options, &fixed);
710
return throw_bounds_warning(session, fixed, var->flags & PLUGIN_VAR_UNSIGNED,
711
var->name, (int64_t) tmp);
714
static int check_func_str(Session *session, drizzle_sys_var *,
715
void *save, drizzle_value *value)
717
char buff[STRING_BUFFER_USUAL_SIZE];
721
length= sizeof(buff);
722
if ((str= value->val_str(value, buff, &length)))
723
str= session->strmake(str, length);
724
*(const char**)save= str;
729
static void update_func_bool(Session *, drizzle_sys_var *,
730
void *tgt, const void *save)
732
*(bool *) tgt= *(int *) save ? 1 : 0;
736
static void update_func_int(Session *, drizzle_sys_var *,
737
void *tgt, const void *save)
739
*(int *)tgt= *(int *) save;
743
static void update_func_long(Session *, drizzle_sys_var *,
744
void *tgt, const void *save)
746
*(long *)tgt= *(long *) save;
750
static void update_func_int64_t(Session *, drizzle_sys_var *,
751
void *tgt, const void *save)
753
*(int64_t *)tgt= *(uint64_t *) save;
757
static void update_func_str(Session *, drizzle_sys_var *var,
758
void *tgt, const void *save)
760
char *old= *(char **) tgt;
761
*(char **)tgt= *(char **) save;
762
if (var->flags & PLUGIN_VAR_MEMALLOC)
764
*(char **)tgt= strdup(*(char **) save);
767
* There isn't a _really_ good thing to do here until this whole set_var
768
* mess gets redesigned
771
errmsg_printf(ERRMSG_LVL_ERROR, _("Out of memory."));
507
777
/****************************************************************************
508
778
System Variables support
509
779
****************************************************************************/
782
sys_var *find_sys_var(Session *, const char *str, uint32_t length)
785
sys_var_pluginvar *pi= NULL;
786
module::Module *module;
788
if ((var= intern_find_sys_var(str, length, false)) &&
789
(pi= var->cast_pluginvar()))
791
if (!(module= pi->plugin))
792
var= NULL; /* failed to lock it, it must be uninstalling */
793
else if (module->isInited == false)
800
If the variable exists but the plugin it is associated with is not ready
801
then the intern_plugin_lock did not raise an error, so we do it here.
804
my_error(ER_UNKNOWN_SYSTEM_VARIABLE, MYF(0), (char*) str);
808
static const string make_bookmark_name(const string &plugin, const char *name, int flags)
810
/* Embed the flags into the first char of the string */
811
string varname(1, static_cast<char>(flags & PLUGIN_VAR_TYPEMASK));
812
varname.append(plugin);
813
varname.push_back('_');
814
varname.append(name);
816
for (string::iterator p= varname.begin() + 1; p != varname.end(); ++p)
827
called by register_var, construct_options and test_plugin_options.
828
Returns the 'bookmark' for the named variable.
829
LOCK_system_variables_hash should be at least read locked
831
static st_bookmark *find_bookmark(const string &plugin, const char *name, int flags)
833
st_bookmark *result= NULL;
835
if (!(flags & PLUGIN_VAR_SessionLOCAL))
838
const string varname(make_bookmark_name(plugin, name, flags));
841
result= (st_bookmark*) hash_search(&bookmark_hash,
842
(const unsigned char*) varname.c_str(), varname.size() - 1);
849
returns a bookmark for session-local variables, creating if neccessary.
850
returns null for non session-local variables.
851
Requires that a write lock is obtained on LOCK_system_variables_hash
853
static st_bookmark *register_var(const string &plugin, const char *name,
856
if (!(flags & PLUGIN_VAR_SessionLOCAL))
859
uint32_t size= 0, offset, new_size;
862
switch (flags & PLUGIN_VAR_TYPEMASK) {
863
case PLUGIN_VAR_BOOL:
864
size= ALIGN_SIZE(sizeof(bool));
867
size= ALIGN_SIZE(sizeof(int));
869
case PLUGIN_VAR_LONG:
870
size= ALIGN_SIZE(sizeof(long));
872
case PLUGIN_VAR_LONGLONG:
873
size= ALIGN_SIZE(sizeof(uint64_t));
876
size= ALIGN_SIZE(sizeof(char*));
884
if (!(result= find_bookmark(plugin, name, flags)))
886
const string varname(make_bookmark_name(plugin, name, flags));
888
result= static_cast<st_bookmark*>(plugin_mem_root.alloc_root(sizeof(struct st_bookmark) + varname.size() + 1));
889
memset(result->key, 0, varname.size()+1);
890
memcpy(result->key, varname.c_str(), varname.size());
891
result->name_len= varname.size() - 2;
894
assert(size && !(size & (size-1))); /* must be power of 2 */
896
offset= global_system_variables.dynamic_variables_size;
897
offset= (offset + size - 1) & ~(size - 1);
898
result->offset= (int) offset;
900
new_size= (offset + size + 63) & ~63;
902
if (new_size > global_variables_dynamic_size)
906
(char *)realloc(global_system_variables.dynamic_variables_ptr,
909
global_system_variables.dynamic_variables_ptr= tmpptr;
912
(char *)realloc(max_system_variables.dynamic_variables_ptr,
915
max_system_variables.dynamic_variables_ptr= tmpptr;
918
Clear the new variable value space. This is required for string
919
variables. If their value is non-NULL, it must point to a valid
922
memset(global_system_variables.dynamic_variables_ptr +
923
global_variables_dynamic_size, 0,
924
new_size - global_variables_dynamic_size);
925
memset(max_system_variables.dynamic_variables_ptr +
926
global_variables_dynamic_size, 0,
927
new_size - global_variables_dynamic_size);
928
global_variables_dynamic_size= new_size;
931
global_system_variables.dynamic_variables_head= offset;
932
max_system_variables.dynamic_variables_head= offset;
933
global_system_variables.dynamic_variables_size= offset + size;
934
max_system_variables.dynamic_variables_size= offset + size;
935
global_system_variables.dynamic_variables_version++;
936
max_system_variables.dynamic_variables_version++;
938
result->version= global_system_variables.dynamic_variables_version;
940
/* this should succeed because we have already checked if a dup exists */
941
if (my_hash_insert(&bookmark_hash, (unsigned char*) result))
943
fprintf(stderr, "failed to add placeholder to hash");
952
returns a pointer to the memory which holds the session-local variable or
953
a pointer to the global variable if session==null.
954
If required, will sync with global variables if the requested variable
955
has not yet been allocated in the current thread.
957
static unsigned char *intern_sys_var_ptr(Session* session, int offset, bool global_lock)
960
assert((uint32_t)offset <= global_system_variables.dynamic_variables_head);
963
return (unsigned char*) global_system_variables.dynamic_variables_ptr + offset;
966
dynamic_variables_head points to the largest valid offset
968
if (!session->variables.dynamic_variables_ptr ||
969
(uint32_t)offset > session->variables.dynamic_variables_head)
974
if (!(tmpptr= (char *)realloc(session->variables.dynamic_variables_ptr,
975
global_variables_dynamic_size)))
977
session->variables.dynamic_variables_ptr= tmpptr;
980
LOCK_global_system_variables.lock();
982
//safe_mutex_assert_owner(&LOCK_global_system_variables);
984
memcpy(session->variables.dynamic_variables_ptr +
985
session->variables.dynamic_variables_size,
986
global_system_variables.dynamic_variables_ptr +
987
session->variables.dynamic_variables_size,
988
global_system_variables.dynamic_variables_size -
989
session->variables.dynamic_variables_size);
992
now we need to iterate through any newly copied 'defaults'
993
and if it is a string type with MEMALLOC flag, we need to strdup
995
for (idx= 0; idx < bookmark_hash.records; idx++)
997
sys_var_pluginvar *pi;
999
st_bookmark *v= (st_bookmark*) hash_element(&bookmark_hash,idx);
1001
if (v->version <= session->variables.dynamic_variables_version ||
1002
!(var= intern_find_sys_var(v->key + 1, v->name_len, true)) ||
1003
!(pi= var->cast_pluginvar()) ||
1004
v->key[0] != (pi->plugin_var->flags & PLUGIN_VAR_TYPEMASK))
1007
/* Here we do anything special that may be required of the data types */
1009
if ((pi->plugin_var->flags & PLUGIN_VAR_TYPEMASK) == PLUGIN_VAR_STR &&
1010
pi->plugin_var->flags & PLUGIN_VAR_MEMALLOC)
1012
char **pp= (char**) (session->variables.dynamic_variables_ptr +
1013
*(int*)(pi->plugin_var + 1));
1014
if ((*pp= *(char**) (global_system_variables.dynamic_variables_ptr +
1015
*(int*)(pi->plugin_var + 1))))
1023
LOCK_global_system_variables.unlock();
1025
session->variables.dynamic_variables_version=
1026
global_system_variables.dynamic_variables_version;
1027
session->variables.dynamic_variables_head=
1028
global_system_variables.dynamic_variables_head;
1029
session->variables.dynamic_variables_size=
1030
global_system_variables.dynamic_variables_size;
1032
return (unsigned char*)session->variables.dynamic_variables_ptr + offset;
1035
static bool *mysql_sys_var_ptr_bool(Session* a_session, int offset)
1037
return (bool *)intern_sys_var_ptr(a_session, offset, true);
1040
static int *mysql_sys_var_ptr_int(Session* a_session, int offset)
1042
return (int *)intern_sys_var_ptr(a_session, offset, true);
1045
static long *mysql_sys_var_ptr_long(Session* a_session, int offset)
1047
return (long *)intern_sys_var_ptr(a_session, offset, true);
1050
static int64_t *mysql_sys_var_ptr_int64_t(Session* a_session, int offset)
1052
return (int64_t *)intern_sys_var_ptr(a_session, offset, true);
1055
static char **mysql_sys_var_ptr_str(Session* a_session, int offset)
1057
return (char **)intern_sys_var_ptr(a_session, offset, true);
513
1060
void plugin_sessionvar_init(Session *session)
515
1062
session->variables.storage_engine= NULL;
516
cleanup_variables(&session->variables);
1063
cleanup_variables(session, &session->variables);
518
1065
session->variables= global_system_variables;
519
1066
session->variables.storage_engine= NULL;
556
1128
void plugin_sessionvar_cleanup(Session *session)
558
1130
unlock_variables(session, &session->variables);
559
cleanup_variables(&session->variables);
1131
cleanup_variables(session, &session->variables);
1136
@brief Free values of thread variables of a plugin.
1138
This must be called before a plugin is deleted. Otherwise its
1139
variables are no longer accessible and the value space is lost. Note
1140
that only string values with PLUGIN_VAR_MEMALLOC are allocated and
1143
@param[in] vars Chain of system variables of a plugin
1146
static void plugin_vars_free_values(sys_var *vars)
1149
for (sys_var *var= vars; var; var= var->getNext())
1151
sys_var_pluginvar *piv= var->cast_pluginvar();
1153
((piv->plugin_var->flags & PLUGIN_VAR_TYPEMASK) == PLUGIN_VAR_STR) &&
1154
(piv->plugin_var->flags & PLUGIN_VAR_MEMALLOC))
1156
/* Free the string from global_system_variables. */
1157
char **valptr= (char**) piv->real_value_ptr(NULL, OPT_GLOBAL);
1166
bool sys_var_pluginvar::check_update_type(Item_result type)
1170
switch (plugin_var->flags & PLUGIN_VAR_TYPEMASK) {
1171
case PLUGIN_VAR_INT:
1172
case PLUGIN_VAR_LONG:
1173
case PLUGIN_VAR_LONGLONG:
1174
return type != INT_RESULT;
1175
case PLUGIN_VAR_STR:
1176
return type != STRING_RESULT;
1183
SHOW_TYPE sys_var_pluginvar::show_type()
1185
switch (plugin_var->flags & PLUGIN_VAR_TYPEMASK) {
1186
case PLUGIN_VAR_BOOL:
1187
return SHOW_MY_BOOL;
1188
case PLUGIN_VAR_INT:
1190
case PLUGIN_VAR_LONG:
1192
case PLUGIN_VAR_LONGLONG:
1193
return SHOW_LONGLONG;
1194
case PLUGIN_VAR_STR:
1195
return SHOW_CHAR_PTR;
1203
unsigned char* sys_var_pluginvar::real_value_ptr(Session *session, sql_var_t type)
1205
assert(session || (type == OPT_GLOBAL));
1206
if (plugin_var->flags & PLUGIN_VAR_SessionLOCAL)
1208
if (type == OPT_GLOBAL)
1211
return intern_sys_var_ptr(session, *(int*) (plugin_var+1), false);
1213
return *(unsigned char**) (plugin_var+1);
1217
TYPELIB* sys_var_pluginvar::plugin_var_typelib(void)
1219
switch (plugin_var->flags & (PLUGIN_VAR_TYPEMASK | PLUGIN_VAR_SessionLOCAL)) {
1220
case PLUGIN_VAR_SessionLOCAL:
1221
return ((sessionvar_enum_t *)plugin_var)->typelib;
1229
unsigned char* sys_var_pluginvar::value_ptr(Session *session, sql_var_t type, const LEX_STRING *)
1231
unsigned char* result;
1233
result= real_value_ptr(session, type);
1239
bool sys_var_pluginvar::check(Session *session, set_var *var)
1241
st_item_value_holder value;
1242
assert(is_readonly() || plugin_var->check);
1244
value.value_type= item_value_type;
1245
value.val_str= item_val_str;
1246
value.val_int= item_val_int;
1247
value.val_real= item_val_real;
1248
value.item= var->value;
1250
return is_readonly() ||
1251
plugin_var->check(session, plugin_var, &var->save_result, &value);
1255
void sys_var_pluginvar::set_default(Session *session, sql_var_t type)
1260
assert(is_readonly() || plugin_var->update);
1265
LOCK_global_system_variables.lock();
1266
tgt= real_value_ptr(session, type);
1267
src= ((void **) (plugin_var + 1) + 1);
1269
if (plugin_var->flags & PLUGIN_VAR_SessionLOCAL)
1271
if (type != OPT_GLOBAL)
1272
src= real_value_ptr(session, OPT_GLOBAL);
1274
switch (plugin_var->flags & PLUGIN_VAR_TYPEMASK) {
1275
case PLUGIN_VAR_INT:
1276
src= &((sessionvar_uint_t*) plugin_var)->def_val;
1278
case PLUGIN_VAR_LONG:
1279
src= &((sessionvar_ulong_t*) plugin_var)->def_val;
1281
case PLUGIN_VAR_LONGLONG:
1282
src= &((sessionvar_uint64_t_t*) plugin_var)->def_val;
1284
case PLUGIN_VAR_BOOL:
1285
src= &((sessionvar_bool_t*) plugin_var)->def_val;
1287
case PLUGIN_VAR_STR:
1288
src= &((sessionvar_str_t*) plugin_var)->def_val;
1295
/* session must equal current_session if PLUGIN_VAR_SessionLOCAL flag is set */
1296
assert(!(plugin_var->flags & PLUGIN_VAR_SessionLOCAL) ||
1297
session == current_session);
1299
if (!(plugin_var->flags & PLUGIN_VAR_SessionLOCAL) || type == OPT_GLOBAL)
1301
plugin_var->update(session, plugin_var, tgt, src);
1302
LOCK_global_system_variables.unlock();
1306
LOCK_global_system_variables.unlock();
1307
plugin_var->update(session, plugin_var, tgt, src);
1312
bool sys_var_pluginvar::update(Session *session, set_var *var)
1316
assert(is_readonly() || plugin_var->update);
1318
/* session must equal current_session if PLUGIN_VAR_SessionLOCAL flag is set */
1319
assert(!(plugin_var->flags & PLUGIN_VAR_SessionLOCAL) ||
1320
session == current_session);
1325
LOCK_global_system_variables.lock();
1326
tgt= real_value_ptr(session, var->type);
1328
if (!(plugin_var->flags & PLUGIN_VAR_SessionLOCAL) || var->type == OPT_GLOBAL)
1330
/* variable we are updating has global scope, so we unlock after updating */
1331
plugin_var->update(session, plugin_var, tgt, &var->save_result);
1332
LOCK_global_system_variables.unlock();
1336
LOCK_global_system_variables.unlock();
1337
plugin_var->update(session, plugin_var, tgt, &var->save_result);
1343
#define OPTION_SET_LIMITS(type, options, opt) \
1344
options->var_type= type; \
1345
options->def_value= (opt)->def_val; \
1346
options->min_value= (opt)->min_val; \
1347
options->max_value= (opt)->max_val; \
1348
options->block_size= (long) (opt)->blk_sz
1351
void plugin_opt_set_limits(struct option *options,
1352
const drizzle_sys_var *opt)
1354
options->sub_size= 0;
1356
switch (opt->flags & (PLUGIN_VAR_TYPEMASK |
1357
PLUGIN_VAR_UNSIGNED | PLUGIN_VAR_SessionLOCAL)) {
1358
/* global system variables */
1359
case PLUGIN_VAR_INT:
1360
OPTION_SET_LIMITS(GET_INT, options, (sysvar_int_t*) opt);
1362
case PLUGIN_VAR_INT | PLUGIN_VAR_UNSIGNED:
1363
OPTION_SET_LIMITS(GET_UINT, options, (sysvar_uint_t*) opt);
1365
case PLUGIN_VAR_LONG:
1366
OPTION_SET_LIMITS(GET_LONG, options, (sysvar_long_t*) opt);
1368
case PLUGIN_VAR_LONG | PLUGIN_VAR_UNSIGNED:
1369
OPTION_SET_LIMITS(GET_ULONG_IS_FAIL, options, (sysvar_ulong_t*) opt);
1371
case PLUGIN_VAR_LONGLONG:
1372
OPTION_SET_LIMITS(GET_LL, options, (sysvar_int64_t_t*) opt);
1374
case PLUGIN_VAR_LONGLONG | PLUGIN_VAR_UNSIGNED:
1375
OPTION_SET_LIMITS(GET_ULL, options, (sysvar_uint64_t_t*) opt);
1377
case PLUGIN_VAR_BOOL:
1378
options->var_type= GET_BOOL;
1379
options->def_value= ((sysvar_bool_t*) opt)->def_val;
1381
case PLUGIN_VAR_STR:
1382
options->var_type= ((opt->flags & PLUGIN_VAR_MEMALLOC) ?
1383
GET_STR_ALLOC : GET_STR);
1384
options->def_value= (intptr_t) ((sysvar_str_t*) opt)->def_val;
1386
/* threadlocal variables */
1387
case PLUGIN_VAR_INT | PLUGIN_VAR_SessionLOCAL:
1388
OPTION_SET_LIMITS(GET_INT, options, (sessionvar_int_t*) opt);
1390
case PLUGIN_VAR_INT | PLUGIN_VAR_UNSIGNED | PLUGIN_VAR_SessionLOCAL:
1391
OPTION_SET_LIMITS(GET_UINT, options, (sessionvar_uint_t*) opt);
1393
case PLUGIN_VAR_LONG | PLUGIN_VAR_SessionLOCAL:
1394
OPTION_SET_LIMITS(GET_LONG, options, (sessionvar_long_t*) opt);
1396
case PLUGIN_VAR_LONG | PLUGIN_VAR_UNSIGNED | PLUGIN_VAR_SessionLOCAL:
1397
OPTION_SET_LIMITS(GET_ULONG_IS_FAIL, options, (sessionvar_ulong_t*) opt);
1399
case PLUGIN_VAR_LONGLONG | PLUGIN_VAR_SessionLOCAL:
1400
OPTION_SET_LIMITS(GET_LL, options, (sessionvar_int64_t_t*) opt);
1402
case PLUGIN_VAR_LONGLONG | PLUGIN_VAR_UNSIGNED | PLUGIN_VAR_SessionLOCAL:
1403
OPTION_SET_LIMITS(GET_ULL, options, (sessionvar_uint64_t_t*) opt);
1405
case PLUGIN_VAR_BOOL | PLUGIN_VAR_SessionLOCAL:
1406
options->var_type= GET_BOOL;
1407
options->def_value= ((sessionvar_bool_t*) opt)->def_val;
1409
case PLUGIN_VAR_STR | PLUGIN_VAR_SessionLOCAL:
1410
options->var_type= ((opt->flags & PLUGIN_VAR_MEMALLOC) ?
1411
GET_STR_ALLOC : GET_STR);
1412
options->def_value= (intptr_t) ((sessionvar_str_t*) opt)->def_val;
1417
options->arg_type= REQUIRED_ARG;
1418
if (opt->flags & PLUGIN_VAR_NOCMDARG)
1419
options->arg_type= NO_ARG;
1420
if (opt->flags & PLUGIN_VAR_OPCMDARG)
1421
options->arg_type= OPT_ARG;
1424
static int get_one_plugin_option(int, const struct option *, char *)
1430
static int construct_options(memory::Root *mem_root, module::Module *tmp,
1434
int localoptionid= 256;
1435
const string plugin_name(tmp->getManifest().name);
1437
size_t namelen= plugin_name.size(), optnamelen;
1440
int index= 0, offset= 0;
1441
drizzle_sys_var *opt, **plugin_option;
1444
string name(plugin_name);
1445
transform(name.begin(), name.end(), name.begin(), ::tolower);
1447
for (string::iterator iter= name.begin(); iter != name.end(); ++iter)
1454
Two passes as the 2nd pass will take pointer addresses for use
1455
by my_getopt and register_var() in the first pass uses realloc
1458
for (plugin_option= tmp->getManifest().system_vars;
1459
plugin_option && *plugin_option; plugin_option++, index++)
1461
opt= *plugin_option;
1462
if (!(opt->flags & PLUGIN_VAR_SessionLOCAL))
1464
if (!(register_var(name, opt->name, opt->flags)))
1466
switch (opt->flags & PLUGIN_VAR_TYPEMASK) {
1467
case PLUGIN_VAR_BOOL:
1468
(((sessionvar_bool_t *)opt)->resolve)= mysql_sys_var_ptr_bool;
1470
case PLUGIN_VAR_INT:
1471
(((sessionvar_int_t *)opt)->resolve)= mysql_sys_var_ptr_int;
1473
case PLUGIN_VAR_LONG:
1474
(((sessionvar_long_t *)opt)->resolve)= mysql_sys_var_ptr_long;
1476
case PLUGIN_VAR_LONGLONG:
1477
(((sessionvar_int64_t_t *)opt)->resolve)= mysql_sys_var_ptr_int64_t;
1479
case PLUGIN_VAR_STR:
1480
(((sessionvar_str_t *)opt)->resolve)= mysql_sys_var_ptr_str;
1483
errmsg_printf(ERRMSG_LVL_ERROR, _("Unknown variable type code 0x%x in plugin '%s'."),
1484
opt->flags, plugin_name.c_str());
1489
for (plugin_option= tmp->getManifest().system_vars;
1490
plugin_option && *plugin_option; plugin_option++, index++)
1492
switch ((opt= *plugin_option)->flags & PLUGIN_VAR_TYPEMASK) {
1493
case PLUGIN_VAR_BOOL:
1495
opt->check= check_func_bool;
1497
opt->update= update_func_bool;
1499
case PLUGIN_VAR_INT:
1501
opt->check= check_func_int;
1503
opt->update= update_func_int;
1505
case PLUGIN_VAR_LONG:
1507
opt->check= check_func_long;
1509
opt->update= update_func_long;
1511
case PLUGIN_VAR_LONGLONG:
1513
opt->check= check_func_int64_t;
1515
opt->update= update_func_int64_t;
1517
case PLUGIN_VAR_STR:
1519
opt->check= check_func_str;
1522
opt->update= update_func_str;
1523
if ((opt->flags & (PLUGIN_VAR_MEMALLOC | PLUGIN_VAR_READONLY)) == false)
1525
opt->flags|= PLUGIN_VAR_READONLY;
1526
errmsg_printf(ERRMSG_LVL_WARN, _("Server variable %s of plugin %s was forced "
1527
"to be read-only: string variable without "
1528
"update_func and PLUGIN_VAR_MEMALLOC flag"),
1529
opt->name, plugin_name.c_str());
1534
errmsg_printf(ERRMSG_LVL_ERROR, _("Unknown variable type code 0x%x in plugin '%s'."),
1535
opt->flags, plugin_name.c_str());
1539
if ((opt->flags & (PLUGIN_VAR_NOCMDOPT | PLUGIN_VAR_SessionLOCAL))
1540
== PLUGIN_VAR_NOCMDOPT)
1545
errmsg_printf(ERRMSG_LVL_ERROR, _("Missing variable name in plugin '%s'."),
1546
plugin_name.c_str());
1550
if (!(opt->flags & PLUGIN_VAR_SessionLOCAL))
1552
optnamelen= strlen(opt->name);
1553
optname= (char*) mem_root->alloc_root(namelen + optnamelen + 2);
1554
sprintf(optname, "%s-%s", name.c_str(), opt->name);
1555
optnamelen= namelen + optnamelen + 1;
1559
/* this should not fail because register_var should create entry */
1560
if (!(v= find_bookmark(name, opt->name, opt->flags)))
1562
errmsg_printf(ERRMSG_LVL_ERROR, _("Thread local variable '%s' not allocated "
1563
"in plugin '%s'."), opt->name, plugin_name.c_str());
1567
*(int*)(opt + 1)= offset= v->offset;
1569
if (opt->flags & PLUGIN_VAR_NOCMDOPT)
1572
optname= (char*) mem_root->memdup_root(v->key + 1, (optnamelen= v->name_len) + 1);
1575
/* convert '_' to '-' */
1576
for (p= optname; *p; p++)
1580
options->name= optname;
1581
options->comment= opt->comment;
1582
options->app_type= opt;
1583
options->id= localoptionid++;
1585
plugin_opt_set_limits(options, opt);
1587
if (opt->flags & PLUGIN_VAR_SessionLOCAL)
1588
options->value= options->u_max_value= (char**)
1589
(global_system_variables.dynamic_variables_ptr + offset);
1591
options->value= options->u_max_value= *(char***) (opt + 1);
1600
static option *construct_help_options(memory::Root *mem_root, module::Module *p)
1602
drizzle_sys_var **opt;
1604
uint32_t count= EXTRA_OPTIONS;
1606
for (opt= p->getManifest().system_vars; opt && *opt; opt++, count++) {};
1608
opts= (option*)mem_root->alloc_root((sizeof(option) * count));
1612
memset(opts, 0, sizeof(option) * count);
1614
if (construct_options(mem_root, p, opts))
1620
void drizzle_add_plugin_sysvar(sys_var_pluginvar *var)
1622
plugin_sysvar_vec.push_back(var);
1625
void drizzle_del_plugin_sysvar()
1627
vector<sys_var_pluginvar *>::iterator iter= plugin_sysvar_vec.begin();
1628
while(iter != plugin_sysvar_vec.end())
1633
plugin_sysvar_vec.clear();
566
1638
test_plugin_options()
567
1639
tmp_root temporary scratch space
568
1640
plugin internal plugin structure
1641
argc user supplied arguments
1642
argv user supplied arguments
569
1643
default_enabled default plugin enable status
571
1645
0 SUCCESS - plugin should be enabled/loaded
573
1647
Requires that a write-lock is held on LOCK_system_variables_hash
575
static int test_plugin_options(memory::Root *,
1649
static int test_plugin_options(memory::Root *module_root,
576
1650
module::Module *test_module,
1651
int *argc, char **argv,
577
1652
po::options_description &long_options)
1654
struct sys_var_chain chain= { NULL, NULL };
1655
drizzle_sys_var **opt;
1659
struct st_bookmark *var;
1660
uint32_t len, count= EXTRA_OPTIONS;
580
1662
if (test_module->getManifest().init_options != NULL)