13
13
along with this program; if not, write to the Free Software
14
14
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
25
#include "drizzled/my_getopt.h"
26
#include "drizzled/my_hash.h"
27
#include "drizzled/internal/m_string.h"
29
#include "drizzled/plugin.h"
30
#include "drizzled/plugin/load_list.h"
31
#include "drizzled/sql_parse.h"
32
#include "drizzled/show.h"
33
#include "drizzled/cursor.h"
34
#include "drizzled/set_var.h"
35
#include "drizzled/session.h"
36
#include "drizzled/item/null.h"
37
#include "drizzled/plugin/registry.h"
38
#include "drizzled/error.h"
39
#include "drizzled/gettext.h"
40
#include "drizzled/errmsg_print.h"
41
#include "drizzled/plugin/library.h"
42
#include "drizzled/strfunc.h"
43
#include "drizzled/pthread_globals.h"
44
#include "drizzled/util/tokenize.h"
46
/* FreeBSD 2.2.2 does not define RTLD_NOW) */
53
typedef drizzled::plugin::Manifest drizzled_builtin_plugin[];
54
extern drizzled_builtin_plugin PANDORA_BUILTIN_LIST;
55
static drizzled::plugin::Manifest *drizzled_builtins[]=
57
PANDORA_BUILTIN_LIST, NULL
64
class sys_var_pluginvar;
65
static vector<sys_var_pluginvar *> plugin_sysvar_vec;
67
char *opt_plugin_add= NULL;
68
char *opt_plugin_remove= NULL;
16
#include "mysql_priv.h"
17
#include <my_pthread.h>
18
#include <my_getopt.h>
19
#define REPORT_TO_LOG 1
20
#define REPORT_TO_USER 2
22
#define plugin_ref_to_int(A) (A ? A[0] : NULL)
23
#define plugin_int_to_ref(A) &(A)
25
extern struct st_mysql_plugin *mysqld_builtins[];
69
27
char *opt_plugin_load= NULL;
70
28
char *opt_plugin_dir_ptr;
71
29
char opt_plugin_dir[FN_REFLEN];
72
const char *opt_plugin_load_default= PANDORA_PLUGIN_LIST;
31
When you ad a new plugin type, add both a string and make sure that the
32
init and deinit array are correctly updated.
34
const LEX_STRING plugin_type_names[MYSQL_MAX_PLUGIN_TYPE_NUM]=
36
{ C_STRING_WITH_LEN("DAEMON") },
37
{ C_STRING_WITH_LEN("STORAGE ENGINE") },
38
{ C_STRING_WITH_LEN("INFORMATION SCHEMA") },
39
{ C_STRING_WITH_LEN("UDF") },
40
{ C_STRING_WITH_LEN("UDA") },
41
{ C_STRING_WITH_LEN("AUDIT") },
42
{ C_STRING_WITH_LEN("LOGGER") },
43
{ C_STRING_WITH_LEN("AUTH") }
46
extern int initialize_schema_table(st_plugin_int *plugin);
47
extern int finalize_schema_table(st_plugin_int *plugin);
49
extern int initialize_udf(st_plugin_int *plugin);
50
extern int finalize_udf(st_plugin_int *plugin);
53
The number of elements in both plugin_type_initialize and
54
plugin_type_deinitialize should equal to the number of plugins
57
plugin_type_init plugin_type_initialize[MYSQL_MAX_PLUGIN_TYPE_NUM]=
60
ha_initialize_handlerton, /* Storage Engine */
61
initialize_schema_table, /* Information Schema */
62
initialize_udf, /* UDF */
69
plugin_type_init plugin_type_deinitialize[MYSQL_MAX_PLUGIN_TYPE_NUM]=
72
ha_finalize_handlerton, /* Storage Engine */
73
finalize_schema_table, /* Information Schema */
74
finalize_udf, /* UDF */
81
static const char *plugin_declarations_sym= "_mysql_plugin_declarations_";
74
83
/* Note that 'int version' must be the first field of every plugin
75
84
sub-structure (plugin->info).
78
static bool initialized= false;
87
static bool initialized= 0;
89
static DYNAMIC_ARRAY plugin_dl_array;
90
static DYNAMIC_ARRAY plugin_array;
91
static HASH plugin_hash[MYSQL_MAX_PLUGIN_TYPE_NUM];
81
92
static bool reap_needed= false;
93
static int plugin_array_version=0;
84
96
write-lock on LOCK_system_variables_hash is required before modifying
85
97
the following variables/structures
87
static memory::Root plugin_mem_root;
88
static uint32_t global_variables_dynamic_size= 0;
99
static MEM_ROOT plugin_mem_root;
100
static uint global_variables_dynamic_size= 0;
89
101
static HASH bookmark_hash;
113
125
struct st_bookmark
135
skeleton of a plugin variable - portion of structure common to all.
137
struct st_mysql_sys_var
139
MYSQL_PLUGIN_VAR_HEADER;
123
144
sys_var class for access to all plugin variables visible to the user
125
146
class sys_var_pluginvar: public sys_var
128
plugin::Module *plugin;
129
drizzle_sys_var *plugin_var;
131
sys_var_pluginvar(const std::string name_arg,
132
drizzle_sys_var *plugin_var_arg)
149
struct st_plugin_int *plugin;
150
struct st_mysql_sys_var *plugin_var;
152
static void *operator new(size_t size, MEM_ROOT *mem_root)
153
{ return (void*) alloc_root(mem_root, (uint) size); }
154
static void operator delete(void *ptr_arg __attribute__((__unused__)),
155
size_t size __attribute__((__unused__)))
156
{ TRASH(ptr_arg, size); }
158
sys_var_pluginvar(const char *name_arg,
159
struct st_mysql_sys_var *plugin_var_arg)
133
160
:sys_var(name_arg), plugin_var(plugin_var_arg) {}
134
161
sys_var_pluginvar *cast_pluginvar() { return this; }
135
162
bool is_readonly() const { return plugin_var->flags & PLUGIN_VAR_READONLY; }
136
bool check_type(sql_var_t type)
137
{ return !(plugin_var->flags & PLUGIN_VAR_SessionLOCAL) && type != OPT_GLOBAL; }
163
bool check_type(enum_var_type type)
164
{ return !(plugin_var->flags & PLUGIN_VAR_THDLOCAL) && type != OPT_GLOBAL; }
138
165
bool check_update_type(Item_result type);
139
166
SHOW_TYPE show_type();
140
unsigned char* real_value_ptr(Session *session, sql_var_t type);
167
uchar* real_value_ptr(THD *thd, enum_var_type type);
141
168
TYPELIB* plugin_var_typelib(void);
142
unsigned char* value_ptr(Session *session, sql_var_t type,
143
const LEX_STRING *base);
144
bool check(Session *session, set_var *var);
145
bool check_default(sql_var_t)
169
uchar* value_ptr(THD *thd, enum_var_type type, LEX_STRING *base);
170
bool check(THD *thd, set_var *var);
171
bool check_default(enum_var_type type __attribute__((__unused__)))
146
172
{ return is_readonly(); }
147
void set_default(Session *session, sql_var_t);
148
bool update(Session *session, set_var *var);
173
void set_default(THD *thd,
174
enum_var_type type __attribute__((__unused__)));
175
bool update(THD *thd, set_var *var);
153
static void plugin_prune_list(vector<string> &plugin_list,
154
const vector<string> &plugins_to_remove);
155
static bool plugin_load_list(plugin::Registry ®istry,
156
memory::Root *tmp_root, int *argc, char **argv,
157
const vector<string> &plugin_list);
158
static int test_plugin_options(memory::Root *, plugin::Module *,
180
static bool plugin_load_list(MEM_ROOT *tmp_root, int *argc, char **argv,
182
static int test_plugin_options(MEM_ROOT *, struct st_plugin_int *,
160
static void unlock_variables(Session *session, struct system_variables *vars);
161
static void cleanup_variables(Session *session, struct system_variables *vars);
184
static bool register_builtin(struct st_mysql_plugin *, struct st_plugin_int *,
185
struct st_plugin_int **);
186
static void unlock_variables(THD *thd, struct system_variables *vars);
187
static void cleanup_variables(THD *thd, struct system_variables *vars);
162
188
static void plugin_vars_free_values(sys_var *vars);
189
static void plugin_opt_set_limits(struct my_option *options,
190
const struct st_mysql_sys_var *opt);
191
#define my_intern_plugin_lock(A,B) intern_plugin_lock(A,B CALLER_INFO)
192
#define my_intern_plugin_lock_ci(A,B) intern_plugin_lock(A,B ORIG_CALLER_INFO)
193
static plugin_ref intern_plugin_lock(LEX *lex, plugin_ref plugin
195
static void intern_plugin_unlock(LEX *lex, plugin_ref plugin);
196
static void reap_plugins(void);
164
199
/* declared in set_var.cc */
165
extern sys_var *intern_find_sys_var(const char *str, uint32_t length, bool no_error);
166
extern bool throw_bounds_warning(Session *session, bool fixed, bool unsignd,
167
const std::string &name, int64_t val);
169
static bool throw_bounds_warning(Session *session, bool fixed, bool unsignd,
170
const char *name, int64_t val)
172
const std::string name_str(name);
173
return throw_bounds_warning(session, fixed, unsignd, name_str, val);
200
extern sys_var *intern_find_sys_var(const char *str, uint length, bool no_error);
201
extern bool throw_bounds_warning(THD *thd, bool fixed, bool unsignd,
202
const char *name, int64_t val);
176
204
/****************************************************************************
177
205
Value type thunks, allows the C world to play in the C++ world
178
206
****************************************************************************/
180
static int item_value_type(drizzle_value *value)
208
static int item_value_type(struct st_mysql_value *value)
182
210
switch (((st_item_value_holder*)value)->item->result_type()) {
184
return DRIZZLE_VALUE_TYPE_INT;
212
return MYSQL_VALUE_TYPE_INT;
185
213
case REAL_RESULT:
186
return DRIZZLE_VALUE_TYPE_REAL;
214
return MYSQL_VALUE_TYPE_REAL;
188
return DRIZZLE_VALUE_TYPE_STRING;
216
return MYSQL_VALUE_TYPE_STRING;
192
static const char *item_val_str(drizzle_value *value,
220
static const char *item_val_str(struct st_mysql_value *value,
193
221
char *buffer, int *length)
195
223
String str(buffer, *length, system_charset_info), *res;
231
259
Plugin support code
232
260
****************************************************************************/
262
static struct st_plugin_dl *plugin_dl_find(const LEX_STRING *dl)
265
struct st_plugin_dl *tmp;
266
for (i= 0; i < plugin_dl_array.elements; i++)
268
tmp= *dynamic_element(&plugin_dl_array, i, struct st_plugin_dl **);
269
if (tmp->ref_count &&
270
! my_strnncoll(files_charset_info,
271
(const uchar *)dl->str, dl->length,
272
(const uchar *)tmp->dl.str, tmp->dl.length))
278
static st_plugin_dl *plugin_dl_insert_or_reuse(struct st_plugin_dl *plugin_dl)
281
struct st_plugin_dl *tmp;
282
for (i= 0; i < plugin_dl_array.elements; i++)
284
tmp= *dynamic_element(&plugin_dl_array, i, struct st_plugin_dl **);
285
if (! tmp->ref_count)
287
memcpy(tmp, plugin_dl, sizeof(struct st_plugin_dl));
291
if (insert_dynamic(&plugin_dl_array, (uchar*)&plugin_dl))
293
tmp= *dynamic_element(&plugin_dl_array, plugin_dl_array.elements - 1,
294
struct st_plugin_dl **)=
295
(struct st_plugin_dl *) memdup_root(&plugin_mem_root, (uchar*)plugin_dl,
296
sizeof(struct st_plugin_dl));
300
static inline void free_plugin_mem(struct st_plugin_dl *p)
304
my_free(p->dl.str, MYF(MY_ALLOW_ZERO_PTR));
308
static st_plugin_dl *plugin_dl_add(const LEX_STRING *dl, int report)
310
char dlpath[FN_REFLEN];
311
uint plugin_dir_len, dummy_errors, dlpathlen;
312
struct st_plugin_dl *tmp, plugin_dl;
314
plugin_dir_len= strlen(opt_plugin_dir);
316
Ensure that the dll doesn't have a path.
317
This is done to ensure that only approved libraries from the
318
plugin directory are used (to make this even remotely secure).
320
if (my_strchr(files_charset_info, dl->str, dl->str + dl->length, FN_LIBCHAR) ||
321
check_string_char_length((LEX_STRING *) dl, "", NAME_CHAR_LEN,
322
system_charset_info, 1) ||
323
plugin_dir_len + dl->length + 1 >= FN_REFLEN)
325
if (report & REPORT_TO_USER)
326
my_error(ER_UDF_NO_PATHS, MYF(0));
327
if (report & REPORT_TO_LOG)
328
sql_print_error(ER(ER_UDF_NO_PATHS));
331
/* If this dll is already loaded just increase ref_count. */
332
if ((tmp= plugin_dl_find(dl)))
337
bzero(&plugin_dl, sizeof(plugin_dl));
338
/* Compile dll path */
340
strxnmov(dlpath, sizeof(dlpath) - 1, opt_plugin_dir, "/", dl->str, NullS) -
342
plugin_dl.ref_count= 1;
343
/* Open new dll handle */
344
if (!(plugin_dl.handle= dlopen(dlpath, RTLD_NOW)))
346
const char *errmsg=dlerror();
347
if (!strncmp(dlpath, errmsg, dlpathlen))
348
{ // if errmsg starts from dlpath, trim this prefix.
350
if (*errmsg == ':') errmsg++;
351
if (*errmsg == ' ') errmsg++;
353
if (report & REPORT_TO_USER)
354
my_error(ER_CANT_OPEN_LIBRARY, MYF(0), dlpath, errno, errmsg);
355
if (report & REPORT_TO_LOG)
356
sql_print_error(ER(ER_CANT_OPEN_LIBRARY), dlpath, errno, errmsg);
360
/* Find plugin declarations */
361
if (!(sym= dlsym(plugin_dl.handle, plugin_declarations_sym)))
363
free_plugin_mem(&plugin_dl);
364
if (report & REPORT_TO_USER)
365
my_error(ER_CANT_FIND_DL_ENTRY, MYF(0), plugin_declarations_sym);
366
if (report & REPORT_TO_LOG)
367
sql_print_error(ER(ER_CANT_FIND_DL_ENTRY), plugin_declarations_sym);
371
plugin_dl.plugins= (struct st_mysql_plugin *)sym;
373
/* Duplicate and convert dll name */
374
plugin_dl.dl.length= dl->length * files_charset_info->mbmaxlen + 1;
375
if (! (plugin_dl.dl.str= (char*) my_malloc(plugin_dl.dl.length, MYF(0))))
377
free_plugin_mem(&plugin_dl);
378
if (report & REPORT_TO_USER)
379
my_error(ER_OUTOFMEMORY, MYF(0), plugin_dl.dl.length);
380
if (report & REPORT_TO_LOG)
381
sql_print_error(ER(ER_OUTOFMEMORY), plugin_dl.dl.length);
384
plugin_dl.dl.length= copy_and_convert(plugin_dl.dl.str, plugin_dl.dl.length,
385
files_charset_info, dl->str, dl->length, system_charset_info,
387
plugin_dl.dl.str[plugin_dl.dl.length]= 0;
388
/* Add this dll to array */
389
if (! (tmp= plugin_dl_insert_or_reuse(&plugin_dl)))
391
free_plugin_mem(&plugin_dl);
392
if (report & REPORT_TO_USER)
393
my_error(ER_OUTOFMEMORY, MYF(0), sizeof(struct st_plugin_dl));
394
if (report & REPORT_TO_LOG)
395
sql_print_error(ER(ER_OUTOFMEMORY), sizeof(struct st_plugin_dl));
402
static void plugin_dl_del(const LEX_STRING *dl)
406
for (i= 0; i < plugin_dl_array.elements; i++)
408
struct st_plugin_dl *tmp= *dynamic_element(&plugin_dl_array, i,
409
struct st_plugin_dl **);
410
if (tmp->ref_count &&
411
! my_strnncoll(files_charset_info,
412
(const uchar *)dl->str, dl->length,
413
(const uchar *)tmp->dl.str, tmp->dl.length))
415
/* Do not remove this element, unless no other plugin uses this dll. */
416
if (! --tmp->ref_count)
418
free_plugin_mem(tmp);
419
bzero(tmp, sizeof(struct st_plugin_dl));
428
static struct st_plugin_int *plugin_find_internal(const LEX_STRING *name, int type)
434
if (type == MYSQL_ANY_PLUGIN)
436
for (i= 0; i < MYSQL_MAX_PLUGIN_TYPE_NUM; i++)
438
struct st_plugin_int *plugin= (st_plugin_int *)
439
hash_search(&plugin_hash[i], (const uchar *)name->str, name->length);
445
return((st_plugin_int *)
446
hash_search(&plugin_hash[type], (const uchar *)name->str, name->length));
451
static SHOW_COMP_OPTION plugin_status(const LEX_STRING *name, int type)
453
SHOW_COMP_OPTION rc= SHOW_OPTION_NO;
454
struct st_plugin_int *plugin;
455
if ((plugin= plugin_find_internal(name, type)))
457
rc= SHOW_OPTION_DISABLED;
458
if (plugin->state == PLUGIN_IS_READY)
465
bool plugin_is_ready(const LEX_STRING *name, int type)
468
if (plugin_status(name, type) == SHOW_OPTION_YES)
474
SHOW_COMP_OPTION sys_var_have_plugin::get_option()
476
LEX_STRING plugin_name= { (char *) plugin_name_str, plugin_name_len };
477
return plugin_status(&plugin_name, plugin_type);
481
static plugin_ref intern_plugin_lock(LEX *lex, plugin_ref rc CALLER_INFO_PROTO)
483
st_plugin_int *pi= plugin_ref_to_int(rc);
485
if (pi->state & (PLUGIN_IS_READY | PLUGIN_IS_UNINITIALIZED))
489
For debugging, we do an additional malloc which allows the
490
memory manager and/or valgrind to track locked references and
491
double unlocks to aid resolving reference counting.problems.
493
if (!(plugin= (plugin_ref) my_malloc_ci(sizeof(pi), MYF(MY_WME))))
500
insert_dynamic(&lex->plugins, (uchar*)&plugin);
507
plugin_ref plugin_lock(THD *thd, plugin_ref *ptr CALLER_INFO_PROTO)
509
LEX *lex= thd ? thd->lex : 0;
511
rc= my_intern_plugin_lock_ci(lex, *ptr);
516
plugin_ref plugin_lock_by_name(THD *thd, const LEX_STRING *name, int type
519
LEX *lex= thd ? thd->lex : 0;
521
st_plugin_int *plugin;
522
if ((plugin= plugin_find_internal(name, type)))
523
rc= my_intern_plugin_lock_ci(lex, plugin_int_to_ref(plugin));
528
static st_plugin_int *plugin_insert_or_reuse(struct st_plugin_int *plugin)
531
struct st_plugin_int *tmp;
532
for (i= 0; i < plugin_array.elements; i++)
534
tmp= *dynamic_element(&plugin_array, i, struct st_plugin_int **);
535
if (tmp->state == PLUGIN_IS_FREED)
537
memcpy(tmp, plugin, sizeof(struct st_plugin_int));
541
if (insert_dynamic(&plugin_array, (uchar*)&plugin))
543
tmp= *dynamic_element(&plugin_array, plugin_array.elements - 1,
544
struct st_plugin_int **)=
545
(struct st_plugin_int *) memdup_root(&plugin_mem_root, (uchar*)plugin,
546
sizeof(struct st_plugin_int));
239
553
Requires that a write-lock is held on LOCK_system_variables_hash
241
static bool plugin_add(plugin::Registry ®istry, memory::Root *tmp_root,
242
plugin::Library *library,
243
int *argc, char **argv)
555
static bool plugin_add(MEM_ROOT *tmp_root,
556
const LEX_STRING *name, const LEX_STRING *dl,
557
int *argc, char **argv, int report)
248
if (registry.find(library->getName()))
559
struct st_plugin_int tmp;
560
struct st_mysql_plugin *plugin;
561
if (plugin_find_internal(name, MYSQL_ANY_PLUGIN))
250
errmsg_printf(ERRMSG_LVL_WARN, ER(ER_PLUGIN_EXISTS),
251
library->getName().c_str());
563
if (report & REPORT_TO_USER)
564
my_error(ER_UDF_EXISTS, MYF(0), name->str);
565
if (report & REPORT_TO_LOG)
566
sql_print_error(ER(ER_UDF_EXISTS), name->str);
255
plugin::Module *tmp= NULL;
569
/* Clear the whole struct to catch future extensions. */
570
bzero((char*) &tmp, sizeof(tmp));
571
if (! (tmp.plugin_dl= plugin_dl_add(dl, report)))
256
573
/* Find plugin by name */
257
const plugin::Manifest *manifest= library->getManifest();
259
if (registry.find(manifest->name))
261
errmsg_printf(ERRMSG_LVL_ERROR,
262
_("Plugin '%s' contains the name '%s' in its manifest, which "
263
"has already been registered.\n"),
264
library->getName().c_str(),
269
tmp= new (std::nothrow) plugin::Module(manifest, library);
273
if (!test_plugin_options(tmp_root, tmp, argc, argv))
278
errmsg_printf(ERRMSG_LVL_ERROR, ER(ER_CANT_FIND_DL_ENTRY),
279
library->getName().c_str());
284
static void delete_module(plugin::Registry ®istry, plugin::Module *module)
286
plugin::Manifest manifest= module->getManifest();
288
if (module->isInited)
291
manifest.deinit(registry);
574
for (plugin= tmp.plugin_dl->plugins; plugin->name; plugin++)
576
uint name_len= strlen(plugin->name);
577
if (plugin->type >= 0 && plugin->type < MYSQL_MAX_PLUGIN_TYPE_NUM &&
578
! my_strnncoll(system_charset_info,
579
(const uchar *)name->str, name->length,
580
(const uchar *)plugin->name,
583
struct st_plugin_int *tmp_plugin_ptr;
586
tmp.name.str= (char *)plugin->name;
587
tmp.name.length= name_len;
589
tmp.state= PLUGIN_IS_UNINITIALIZED;
590
if (!test_plugin_options(tmp_root, &tmp, argc, argv))
592
if ((tmp_plugin_ptr= plugin_insert_or_reuse(&tmp)))
594
plugin_array_version++;
595
if (!my_hash_insert(&plugin_hash[plugin->type], (uchar*)tmp_plugin_ptr))
597
init_alloc_root(&tmp_plugin_ptr->mem_root, 4096, 4096);
600
tmp_plugin_ptr->state= PLUGIN_IS_FREED;
602
mysql_del_sys_var_chain(tmp.system_vars);
605
/* plugin was disabled */
610
if (report & REPORT_TO_USER)
611
my_error(ER_CANT_FIND_DL_ENTRY, MYF(0), name->str);
612
if (report & REPORT_TO_LOG)
613
sql_print_error(ER(ER_CANT_FIND_DL_ENTRY), name->str);
620
static void plugin_deinitialize(struct st_plugin_int *plugin, bool ref_check)
622
if (plugin->plugin->status_vars)
626
We have a problem right now where we can not prepend without
627
breaking backwards compatibility. We will fix this shortly so
628
that engines have "use names" and we wil use those for
629
CREATE TABLE, and use the plugin name then for adding automatic
633
{plugin->plugin->name, (char*)plugin->plugin->status_vars, SHOW_ARRAY},
636
remove_status_vars(array);
638
remove_status_vars(plugin->plugin->status_vars);
639
#endif /* FIX_LATER */
642
if (plugin_type_deinitialize[plugin->plugin->type])
644
if ((*plugin_type_deinitialize[plugin->plugin->type])(plugin))
646
sql_print_error("Plugin '%s' of type %s failed deinitialization",
647
plugin->name.str, plugin_type_names[plugin->plugin->type].str);
650
else if (plugin->plugin->deinit)
651
plugin->plugin->deinit(plugin);
653
plugin->state= PLUGIN_IS_UNINITIALIZED;
656
We do the check here because NDB has a worker THD which doesn't
657
exit until NDB is shut down.
659
if (ref_check && plugin->ref_count)
660
sql_print_error("Plugin '%s' has ref_count=%d after deinitialization.",
661
plugin->name.str, plugin->ref_count);
665
static void plugin_del(struct st_plugin_int *plugin)
294
667
/* Free allocated strings before deleting the plugin. */
295
plugin_vars_free_values(module->system_vars);
296
module->isInited= false;
297
pthread_rwlock_wrlock(&LOCK_system_variables_hash);
298
mysql_del_sys_var_chain(module->system_vars);
299
pthread_rwlock_unlock(&LOCK_system_variables_hash);
304
static void reap_plugins(plugin::Registry ®istry)
306
plugin::Module *module;
308
std::map<std::string, plugin::Module *>::const_iterator modules=
309
registry.getModulesMap().begin();
311
while (modules != registry.getModulesMap().end())
313
module= (*modules).second;
314
delete_module(registry, module);
317
drizzle_del_plugin_sysvar();
321
static void plugin_initialize_vars(plugin::Module *module)
668
plugin_vars_free_values(plugin->system_vars);
669
hash_delete(&plugin_hash[plugin->plugin->type], (uchar*)plugin);
670
if (plugin->plugin_dl)
671
plugin_dl_del(&plugin->plugin_dl->dl);
672
plugin->state= PLUGIN_IS_FREED;
673
plugin_array_version++;
674
rw_wrlock(&LOCK_system_variables_hash);
675
mysql_del_sys_var_chain(plugin->system_vars);
676
rw_unlock(&LOCK_system_variables_hash);
677
free_root(&plugin->mem_root, MYF(0));
681
static void reap_plugins(void)
684
struct st_plugin_int *plugin, **reap, **list;
690
count= plugin_array.elements;
691
reap= (struct st_plugin_int **)my_alloca(sizeof(plugin)*(count+1));
694
for (idx= 0; idx < count; idx++)
696
plugin= *dynamic_element(&plugin_array, idx, struct st_plugin_int **);
697
if (plugin->state == PLUGIN_IS_DELETED && !plugin->ref_count)
699
/* change the status flag to prevent reaping by another thread */
700
plugin->state= PLUGIN_IS_DYING;
706
while ((plugin= *(--list)))
707
plugin_deinitialize(plugin, true);
709
while ((plugin= *(--reap)))
715
static void intern_plugin_unlock(LEX *lex, plugin_ref plugin)
723
pi= plugin_ref_to_int(plugin);
725
my_free((uchar*) plugin, MYF(MY_WME));
730
Remove one instance of this plugin from the use list.
731
We are searching backwards so that plugins locked last
732
could be unlocked faster - optimizing for LIFO semantics.
734
for (i= lex->plugins.elements - 1; i >= 0; i--)
735
if (plugin == *dynamic_element(&lex->plugins, i, plugin_ref*))
737
delete_dynamic_element(&lex->plugins, i);
743
assert(pi->ref_count);
746
if (pi->state == PLUGIN_IS_DELETED && !pi->ref_count)
753
void plugin_unlock(THD *thd, plugin_ref plugin)
755
LEX *lex= thd ? thd->lex : 0;
758
intern_plugin_unlock(lex, plugin);
763
void plugin_unlock_list(THD *thd, plugin_ref *list, uint count)
765
LEX *lex= thd ? thd->lex : 0;
768
intern_plugin_unlock(lex, *list++);
773
static int plugin_initialize(struct st_plugin_int *plugin)
776
if (plugin_type_initialize[plugin->plugin->type])
778
if ((*plugin_type_initialize[plugin->plugin->type])(plugin))
780
sql_print_error("Plugin '%s' registration as a %s failed.",
781
plugin->name.str, plugin_type_names[plugin->plugin->type].str);
785
else if (plugin->plugin->init)
787
if (plugin->plugin->init(plugin))
789
sql_print_error("Plugin '%s' init function returned error.",
795
plugin->state= PLUGIN_IS_READY;
797
if (plugin->plugin->status_vars)
801
We have a problem right now where we can not prepend without
802
breaking backwards compatibility. We will fix this shortly so
803
that engines have "use names" and we wil use those for
804
CREATE TABLE, and use the plugin name then for adding automatic
808
{plugin->plugin->name, (char*)plugin->plugin->status_vars, SHOW_ARRAY},
811
if (add_status_vars(array)) // add_status_vars makes a copy
814
add_status_vars(plugin->plugin->status_vars); // add_status_vars makes a copy
815
#endif /* FIX_LATER */
324
819
set the plugin attribute of plugin's sys vars so they are pointing
325
820
to the active plugin
327
if (module->system_vars)
822
if (plugin->system_vars)
329
sys_var_pluginvar *var= module->system_vars->cast_pluginvar();
824
sys_var_pluginvar *var= plugin->system_vars->cast_pluginvar();
333
if (! var->getNext())
335
var= var->getNext()->cast_pluginvar();
830
var= var->next->cast_pluginvar();
341
static bool plugin_initialize(plugin::Registry ®istry,
342
plugin::Module *module)
840
extern "C" uchar *get_plugin_hash_key(const uchar *, size_t *, bool);
841
extern "C" uchar *get_bookmark_hash_key(const uchar *, size_t *, bool);
844
uchar *get_plugin_hash_key(const uchar *buff, size_t *length,
845
bool not_used __attribute__((unused)))
344
assert(module->isInited == false);
346
registry.setCurrentModule(module);
347
if (module->getManifest().init)
349
if (module->getManifest().init(registry))
351
errmsg_printf(ERRMSG_LVL_ERROR,
352
_("Plugin '%s' init function returned error.\n"),
353
module->getName().c_str());
357
registry.clearCurrentModule();
358
module->isInited= true;
847
struct st_plugin_int *plugin= (st_plugin_int *)buff;
848
*length= (uint)plugin->name.length;
849
return((uchar *)plugin->name.str);
365
static unsigned char *get_bookmark_hash_key(const unsigned char *buff,
366
size_t *length, bool)
853
uchar *get_bookmark_hash_key(const uchar *buff, size_t *length,
854
bool not_used __attribute__((unused)))
368
856
struct st_bookmark *var= (st_bookmark *)buff;
369
857
*length= var->name_len + 1;
370
return (unsigned char*) var->key;
858
return (uchar*) var->key;
379
867
Finally we initialize everything, aka the dynamic that have yet to initialize.
381
bool plugin_init(plugin::Registry ®istry,
382
int *argc, char **argv,
869
int plugin_init(int *argc, char **argv, int flags)
385
plugin::Manifest **builtins;
386
plugin::Manifest *manifest;
387
plugin::Module *module;
388
memory::Root tmp_root;
872
struct st_mysql_plugin **builtins;
873
struct st_mysql_plugin *plugin;
874
struct st_plugin_int tmp, *plugin_ptr, **reap;
393
init_alloc_root(&plugin_mem_root, 4096);
394
init_alloc_root(&tmp_root, 4096);
880
init_alloc_root(&plugin_mem_root, 4096, 4096);
881
init_alloc_root(&tmp_root, 4096, 4096);
396
883
if (hash_init(&bookmark_hash, &my_charset_bin, 16, 0, 0,
397
884
get_bookmark_hash_key, NULL, HASH_UNIQUE))
888
if (my_init_dynamic_array(&plugin_dl_array,
889
sizeof(struct st_plugin_dl *),16,16) ||
890
my_init_dynamic_array(&plugin_array,
891
sizeof(struct st_plugin_int *),16,16))
894
for (i= 0; i < MYSQL_MAX_PLUGIN_TYPE_NUM; i++)
399
free_root(&tmp_root, MYF(0));
896
if (hash_init(&plugin_hash[i], system_charset_info, 16, 0, 0,
897
get_plugin_hash_key, NULL, HASH_UNIQUE))
407
904
First we register builtin plugins
409
for (builtins= drizzled_builtins; *builtins; builtins++)
906
for (builtins= mysqld_builtins; *builtins; builtins++)
412
if (manifest->name != NULL)
908
for (plugin= *builtins; plugin->name; plugin++)
414
module= new (std::nothrow) plugin::Module(manifest);
910
bzero(&tmp, sizeof(tmp));
912
tmp.name.str= (char *)plugin->name;
913
tmp.name.length= strlen(plugin->name);
418
free_root(&tmp_root, MYF(memory::MARK_BLOCKS_FREE));
419
if (test_plugin_options(&tmp_root, module, argc, argv))
915
free_root(&tmp_root, MYF(MY_MARK_BLOCKS_FREE));
916
if (test_plugin_options(&tmp_root, &tmp, argc, argv))
422
registry.add(module);
424
plugin_initialize_vars(module);
919
if (register_builtin(plugin, &tmp, &plugin_ptr))
922
if (plugin_initialize(plugin_ptr))
926
initialize the global default storage engine so that it may
927
not be null in any child thread.
929
if (my_strcasecmp(&my_charset_latin1, plugin->name, "MyISAM") == 0)
428
if (plugin_initialize(registry, module))
430
free_root(&tmp_root, MYF(0));
931
assert(!global_system_variables.table_plugin);
932
global_system_variables.table_plugin=
933
my_intern_plugin_lock(NULL, plugin_int_to_ref(plugin_ptr));
934
assert(plugin_ptr->ref_count == 1);
438
bool load_failed= false;
439
vector<string> plugin_list;
442
tokenize(opt_plugin_load, plugin_list, ",", true);
446
tokenize(opt_plugin_load_default, plugin_list, ",", true);
450
tokenize(opt_plugin_add, plugin_list, ",", true);
453
if (opt_plugin_remove)
455
vector<string> plugins_to_remove;
456
tokenize(opt_plugin_remove, plugins_to_remove, ",", true);
457
plugin_prune_list(plugin_list, plugins_to_remove);
939
/* should now be set to MyISAM storage engine */
940
assert(global_system_variables.table_plugin);
460
942
/* Register all dynamic plugins */
461
load_failed= plugin_load_list(registry, &tmp_root, argc, argv,
943
if (!(flags & PLUGIN_INIT_SKIP_DYNAMIC_LOADING))
465
free_root(&tmp_root, MYF(0));
946
plugin_load_list(&tmp_root, argc, argv, opt_plugin_load);
471
free_root(&tmp_root, MYF(0));
949
if (flags & PLUGIN_INIT_SKIP_INITIALIZATION)
476
953
Now we initialize all remaining plugins
478
std::map<std::string, plugin::Module *>::const_iterator modules=
479
registry.getModulesMap().begin();
481
while (modules != registry.getModulesMap().end())
956
reap= (st_plugin_int **) my_alloca((plugin_array.elements+1) * sizeof(void*));
959
for (i= 0; i < plugin_array.elements; i++)
483
module= (*modules).second;
485
if (module->isInited == false)
961
plugin_ptr= *dynamic_element(&plugin_array, i, struct st_plugin_int **);
962
if (plugin_ptr->state == PLUGIN_IS_UNINITIALIZED)
487
plugin_initialize_vars(module);
489
if (plugin_initialize(registry, module))
490
delete_module(registry, module);
964
if (plugin_initialize(plugin_ptr))
966
plugin_ptr->state= PLUGIN_IS_DYING;
967
*(reap++)= plugin_ptr;
495
free_root(&tmp_root, MYF(0));
501
public unary_function<string, bool>
503
const string to_match;
505
PrunePlugin& operator=(const PrunePlugin&);
507
explicit PrunePlugin(const string &match_in) :
511
result_type operator()(const string &match_against)
513
return match_against == to_match;
517
static void plugin_prune_list(vector<string> &plugin_list,
518
const vector<string> &plugins_to_remove)
520
for (vector<string>::const_iterator iter= plugins_to_remove.begin();
521
iter != plugins_to_remove.end();
524
plugin_list.erase(remove_if(plugin_list.begin(),
973
Check if any plugins have to be reaped
975
while ((plugin_ptr= *(--reap)))
977
plugin_deinitialize(plugin_ptr, true);
978
plugin_del(plugin_ptr);
984
free_root(&tmp_root, MYF(0));
990
free_root(&tmp_root, MYF(0));
995
static bool register_builtin(struct st_mysql_plugin *plugin,
996
struct st_plugin_int *tmp,
997
struct st_plugin_int **ptr)
1000
tmp->state= PLUGIN_IS_UNINITIALIZED;
1004
if (insert_dynamic(&plugin_array, (uchar*)&tmp))
1007
*ptr= *dynamic_element(&plugin_array, plugin_array.elements - 1,
1008
struct st_plugin_int **)=
1009
(struct st_plugin_int *) memdup_root(&plugin_mem_root, (uchar*)tmp,
1010
sizeof(struct st_plugin_int));
1012
if (my_hash_insert(&plugin_hash[plugin->type],(uchar*) *ptr))
532
1020
called only by plugin_init()
534
static bool plugin_load_list(plugin::Registry ®istry,
535
memory::Root *tmp_root, int *argc, char **argv,
536
const vector<string> &plugin_list)
1022
static bool plugin_load_list(MEM_ROOT *tmp_root, int *argc, char **argv,
538
plugin::Library *library= NULL;
540
for (vector<string>::const_iterator iter= plugin_list.begin();
541
iter != plugin_list.end();
1025
char buffer[FN_REFLEN];
1026
LEX_STRING name= {buffer, 0}, dl= {NULL, 0}, *str= &name;
1027
struct st_plugin_dl *plugin_dl;
1028
struct st_mysql_plugin *plugin;
544
const string plugin_name(*iter);
545
library= registry.addLibrary(plugin_name);
1032
if (p == buffer + sizeof(buffer) - 1)
548
errmsg_printf(ERRMSG_LVL_ERROR,
549
_("Couldn't load plugin library named '%s'.\n"),
550
plugin_name.c_str());
1034
sql_print_error("plugin-load parameter too long");
554
free_root(tmp_root, MYF(memory::MARK_BLOCKS_FREE));
555
if (plugin_add(registry, tmp_root, library, argc, argv))
557
registry.removeLibrary(plugin_name);
558
errmsg_printf(ERRMSG_LVL_ERROR,
559
_("Couldn't load plugin named '%s'.\n"),
560
plugin_name.c_str());
1038
switch ((*(p++)= *(list++))) {
1040
list= NULL; /* terminate the loop */
1042
case ':': /* can't use this as delimiter as it may be drive letter */
1044
str->str[str->length]= '\0';
1045
if (str == &name) // load all plugins in named module
1049
p--; /* reset pointer */
1054
if ((plugin_dl= plugin_dl_add(&dl, REPORT_TO_LOG)))
1056
for (plugin= plugin_dl->plugins; plugin->name; plugin++)
1058
name.str= (char *) plugin->name;
1059
name.length= strlen(name.str);
1061
free_root(tmp_root, MYF(MY_MARK_BLOCKS_FREE));
1062
if (plugin_add(tmp_root, &name, &dl, argc, argv, REPORT_TO_LOG))
1065
plugin_dl_del(&dl); // reduce ref count
1070
free_root(tmp_root, MYF(MY_MARK_BLOCKS_FREE));
1071
if (plugin_add(tmp_root, &name, &dl, argc, argv, REPORT_TO_LOG))
1074
name.length= dl.length= 0;
1075
dl.str= NULL; name.str= p= buffer;
1082
name.str[name.length]= '\0';
1094
sql_print_error("Couldn't load plugin named '%s' with soname '%s'.",
569
void plugin_shutdown(plugin::Registry ®istry)
1100
void plugin_shutdown(void)
1102
uint i, count= plugin_array.elements, free_slots= 0;
1103
struct st_plugin_int **plugins, *plugin;
1104
struct st_plugin_dl **dl;
572
1106
if (initialized)
574
1108
reap_needed= true;
576
reap_plugins(registry);
577
unlock_variables(NULL, &global_system_variables);
578
unlock_variables(NULL, &max_system_variables);
1111
We want to shut down plugins in a reasonable order, this will
1112
become important when we have plugins which depend upon each other.
1113
Circular references cannot be reaped so they are forced afterwards.
1114
TODO: Have an additional step here to notify all active plugins that
1115
shutdown is requested to allow plugins to deinitialize in parallel.
1117
while (reap_needed && (count= plugin_array.elements))
1120
for (i= free_slots= 0; i < count; i++)
1122
plugin= *dynamic_element(&plugin_array, i, struct st_plugin_int **);
1123
switch (plugin->state) {
1124
case PLUGIN_IS_READY:
1125
plugin->state= PLUGIN_IS_DELETED;
1128
case PLUGIN_IS_FREED:
1129
case PLUGIN_IS_UNINITIALIZED:
1137
release any plugin references held.
1139
unlock_variables(NULL, &global_system_variables);
1140
unlock_variables(NULL, &max_system_variables);
1144
if (count > free_slots)
1145
sql_print_warning("Forcing shutdown of %d plugins", count - free_slots);
1147
plugins= (struct st_plugin_int **) my_alloca(sizeof(void*) * (count+1));
1150
If we have any plugins which did not die cleanly, we force shutdown
1152
for (i= 0; i < count; i++)
1154
plugins[i]= *dynamic_element(&plugin_array, i, struct st_plugin_int **);
1155
/* change the state to ensure no reaping races */
1156
if (plugins[i]->state == PLUGIN_IS_DELETED)
1157
plugins[i]->state= PLUGIN_IS_DYING;
1161
We loop through all plugins and call deinit() if they have one.
1163
for (i= 0; i < count; i++)
1164
if (!(plugins[i]->state & (PLUGIN_IS_UNINITIALIZED | PLUGIN_IS_FREED)))
1166
sql_print_information("Plugin '%s' will be forced to shutdown",
1167
plugins[i]->name.str);
1169
We are forcing deinit on plugins so we don't want to do a ref_count
1170
check until we have processed all the plugins.
1172
plugin_deinitialize(plugins[i], false);
1176
We defer checking ref_counts until after all plugins are deinitialized
1177
as some may have worker threads holding on to plugin references.
1179
for (i= 0; i < count; i++)
1181
if (plugins[i]->ref_count)
1182
sql_print_error("Plugin '%s' has ref_count=%d after shutdown.",
1183
plugins[i]->name.str, plugins[i]->ref_count);
1184
if (plugins[i]->state & PLUGIN_IS_UNINITIALIZED)
1185
plugin_del(plugins[i]);
1189
Now we can deallocate all memory.
580
1192
cleanup_variables(NULL, &global_system_variables);
581
1193
cleanup_variables(NULL, &max_system_variables);
586
1200
/* Dispose of the memory */
1202
for (i= 0; i < MYSQL_MAX_PLUGIN_TYPE_NUM; i++)
1203
hash_free(&plugin_hash[i]);
1204
delete_dynamic(&plugin_array);
1206
count= plugin_dl_array.elements;
1207
dl= (struct st_plugin_dl **)my_alloca(sizeof(void*) * count);
1208
for (i= 0; i < count; i++)
1209
dl[i]= *dynamic_element(&plugin_dl_array, i, struct st_plugin_dl **);
1210
for (i= 0; i < plugin_dl_array.elements; i++)
1211
free_plugin_mem(dl[i]);
1213
delete_dynamic(&plugin_dl_array);
588
1215
hash_free(&bookmark_hash);
589
1216
free_root(&plugin_mem_root, MYF(0));
591
1218
global_variables_dynamic_size= 0;
1224
bool plugin_foreach_with_mask(THD *thd, plugin_foreach_func *func,
1225
int type, uint state_mask, void *arg)
1228
struct st_plugin_int *plugin, **plugins;
1229
int version=plugin_array_version;
1234
state_mask= ~state_mask; // do it only once
1236
total= type == MYSQL_ANY_PLUGIN ? plugin_array.elements
1237
: plugin_hash[type].records;
1239
Do the alloca out here in case we do have a working alloca:
1240
leaving the nested stack frame invalidates alloca allocation.
1242
plugins=(struct st_plugin_int **)my_alloca(total*sizeof(plugin));
1243
if (type == MYSQL_ANY_PLUGIN)
1245
for (idx= 0; idx < total; idx++)
1247
plugin= *dynamic_element(&plugin_array, idx, struct st_plugin_int **);
1248
plugins[idx]= !(plugin->state & state_mask) ? plugin : NULL;
1253
HASH *hash= plugin_hash + type;
1254
for (idx= 0; idx < total; idx++)
1256
plugin= (struct st_plugin_int *) hash_element(hash, idx);
1257
plugins[idx]= !(plugin->state & state_mask) ? plugin : NULL;
1260
for (idx= 0; idx < total; idx++)
1262
if (unlikely(version != plugin_array_version))
1264
for (uint i=idx; i < total; i++)
1265
if (plugins[i] && plugins[i]->state & state_mask)
1268
plugin= plugins[idx];
1269
/* It will stop iterating on first engine error when "func" returns true */
1270
if (plugin && func(thd, plugin_int_to_ref(plugin), arg))
594
1282
/****************************************************************************
595
1283
Internal type declarations for variables support
596
1284
****************************************************************************/
598
#undef DRIZZLE_SYSVAR_NAME
599
#define DRIZZLE_SYSVAR_NAME(name) name
1286
#undef MYSQL_SYSVAR_NAME
1287
#define MYSQL_SYSVAR_NAME(name) name
600
1288
#define PLUGIN_VAR_TYPEMASK 0x007f
602
static const uint32_t EXTRA_OPTIONS= 1; /* handle the NULL option */
604
typedef DECLARE_DRIZZLE_SYSVAR_BASIC(sysvar_bool_t, bool);
605
typedef DECLARE_DRIZZLE_SessionVAR_BASIC(sessionvar_bool_t, bool);
606
typedef DECLARE_DRIZZLE_SYSVAR_BASIC(sysvar_str_t, char *);
607
typedef DECLARE_DRIZZLE_SessionVAR_BASIC(sessionvar_str_t, char *);
609
typedef DECLARE_DRIZZLE_SessionVAR_TYPELIB(sessionvar_enum_t, unsigned long);
610
typedef DECLARE_DRIZZLE_SessionVAR_TYPELIB(sessionvar_set_t, uint64_t);
612
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_int_t, int);
613
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_long_t, long);
614
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_int64_t_t, int64_t);
615
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_uint_t, uint);
616
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_ulong_t, ulong);
617
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_uint64_t_t, uint64_t);
619
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_int_t, int);
620
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_long_t, long);
621
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_int64_t_t, int64_t);
622
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_uint_t, uint);
623
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_ulong_t, ulong);
624
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_uint64_t_t, uint64_t);
626
typedef bool *(*mysql_sys_var_ptr_p)(Session* a_session, int offset);
1290
#define EXTRA_OPTIONS 3 /* options for: 'foo', 'plugin-foo' and NULL */
1292
typedef DECLARE_MYSQL_SYSVAR_BASIC(sysvar_bool_t, bool);
1293
typedef DECLARE_MYSQL_THDVAR_BASIC(thdvar_bool_t, bool);
1294
typedef DECLARE_MYSQL_SYSVAR_BASIC(sysvar_str_t, char *);
1295
typedef DECLARE_MYSQL_THDVAR_BASIC(thdvar_str_t, char *);
1297
typedef DECLARE_MYSQL_SYSVAR_TYPELIB(sysvar_enum_t, unsigned long);
1298
typedef DECLARE_MYSQL_THDVAR_TYPELIB(thdvar_enum_t, unsigned long);
1299
typedef DECLARE_MYSQL_SYSVAR_TYPELIB(sysvar_set_t, uint64_t);
1300
typedef DECLARE_MYSQL_THDVAR_TYPELIB(thdvar_set_t, uint64_t);
1302
typedef DECLARE_MYSQL_SYSVAR_SIMPLE(sysvar_int_t, int);
1303
typedef DECLARE_MYSQL_SYSVAR_SIMPLE(sysvar_long_t, long);
1304
typedef DECLARE_MYSQL_SYSVAR_SIMPLE(sysvar_int64_t_t, int64_t);
1305
typedef DECLARE_MYSQL_SYSVAR_SIMPLE(sysvar_uint_t, uint);
1306
typedef DECLARE_MYSQL_SYSVAR_SIMPLE(sysvar_ulong_t, ulong);
1307
typedef DECLARE_MYSQL_SYSVAR_SIMPLE(sysvar_uint64_t_t, uint64_t);
1309
typedef DECLARE_MYSQL_THDVAR_SIMPLE(thdvar_int_t, int);
1310
typedef DECLARE_MYSQL_THDVAR_SIMPLE(thdvar_long_t, long);
1311
typedef DECLARE_MYSQL_THDVAR_SIMPLE(thdvar_int64_t_t, int64_t);
1312
typedef DECLARE_MYSQL_THDVAR_SIMPLE(thdvar_uint_t, uint);
1313
typedef DECLARE_MYSQL_THDVAR_SIMPLE(thdvar_ulong_t, ulong);
1314
typedef DECLARE_MYSQL_THDVAR_SIMPLE(thdvar_uint64_t_t, uint64_t);
1316
typedef bool *(*mysql_sys_var_ptr_p)(THD* a_thd, int offset);
629
1319
/****************************************************************************
630
1320
default variable data check and update functions
631
1321
****************************************************************************/
633
static int check_func_bool(Session *, drizzle_sys_var *var,
634
void *save, drizzle_value *value)
1323
static int check_func_bool(THD *thd __attribute__((__unused__)),
1324
struct st_mysql_sys_var *var,
1325
void *save, st_mysql_value *value)
636
1327
char buff[STRING_BUFFER_USUAL_SIZE];
637
1328
const char *strvalue= "NULL", *str;
638
1329
int result, length;
641
if (value->value_type(value) == DRIZZLE_VALUE_TYPE_STRING)
1332
if (value->value_type(value) == MYSQL_VALUE_TYPE_STRING)
643
1334
length= sizeof(buff);
644
1335
if (!(str= value->val_str(value, buff, &length)) ||
738
1430
length= sizeof(buff);
739
1431
if ((str= value->val_str(value, buff, &length)))
740
str= session->strmake(str, length);
1432
str= thd->strmake(str, length);
741
1433
*(const char**)save= str;
746
static void update_func_bool(Session *, drizzle_sys_var *,
1438
static int check_func_enum(THD *thd __attribute__((__unused__)),
1439
struct st_mysql_sys_var *var,
1440
void *save, st_mysql_value *value)
1442
char buff[STRING_BUFFER_USUAL_SIZE];
1443
const char *strvalue= "NULL", *str;
1449
if (var->flags & PLUGIN_VAR_THDLOCAL)
1450
typelib= ((thdvar_enum_t*) var)->typelib;
1452
typelib= ((sysvar_enum_t*) var)->typelib;
1454
if (value->value_type(value) == MYSQL_VALUE_TYPE_STRING)
1456
length= sizeof(buff);
1457
if (!(str= value->val_str(value, buff, &length)))
1459
if ((result= (long)find_type(typelib, str, length, 1)-1) < 0)
1467
if (value->val_int(value, &tmp))
1469
if (tmp >= typelib->count)
1477
*(long*)save= result;
1480
my_error(ER_WRONG_VALUE_FOR_VAR, MYF(0), var->name, strvalue);
1485
static int check_func_set(THD *thd __attribute__((__unused__)),
1486
struct st_mysql_sys_var *var,
1487
void *save, st_mysql_value *value)
1489
char buff[STRING_BUFFER_USUAL_SIZE], *error= 0;
1490
const char *strvalue= "NULL", *str;
1497
if (var->flags & PLUGIN_VAR_THDLOCAL)
1498
typelib= ((thdvar_set_t*) var)->typelib;
1500
typelib= ((sysvar_set_t*)var)->typelib;
1502
if (value->value_type(value) == MYSQL_VALUE_TYPE_STRING)
1504
length= sizeof(buff);
1505
if (!(str= value->val_str(value, buff, &length)))
1507
result= find_set(typelib, str, length, NULL,
1508
&error, &error_len, ¬_used);
1511
strmake(buff, error, min(sizeof(buff), error_len));
1518
if (value->val_int(value, (int64_t *)&result))
1520
if (unlikely((result >= (1ULL << typelib->count)) &&
1521
(typelib->count < sizeof(long)*8)))
1523
llstr(result, buff);
1528
*(uint64_t*)save= result;
1531
my_error(ER_WRONG_VALUE_FOR_VAR, MYF(0), var->name, strvalue);
1536
static void update_func_bool(THD *thd __attribute__((__unused__)),
1537
struct st_mysql_sys_var *var __attribute__((__unused__)),
747
1538
void *tgt, const void *save)
749
*(bool *) tgt= *(int *) save ? 1 : 0;
1540
*(my_bool *) tgt= *(int *) save ? 1 : 0;
753
static void update_func_int(Session *, drizzle_sys_var *,
1544
static void update_func_int(THD *thd __attribute__((__unused__)),
1545
struct st_mysql_sys_var *var __attribute__((__unused__)),
754
1546
void *tgt, const void *save)
756
1548
*(int *)tgt= *(int *) save;
760
static void update_func_long(Session *, drizzle_sys_var *,
1552
static void update_func_long(THD *thd __attribute__((__unused__)),
1553
struct st_mysql_sys_var *var __attribute__((__unused__)),
761
1554
void *tgt, const void *save)
763
1556
*(long *)tgt= *(long *) save;
767
static void update_func_int64_t(Session *, drizzle_sys_var *,
1560
static void update_func_int64_t(THD *thd __attribute__((__unused__)),
1561
struct st_mysql_sys_var *var __attribute__((__unused__)),
768
1562
void *tgt, const void *save)
770
1564
*(int64_t *)tgt= *(uint64_t *) save;
774
static void update_func_str(Session *, drizzle_sys_var *var,
1568
static void update_func_str(THD *thd __attribute__((__unused__)), struct st_mysql_sys_var *var,
775
1569
void *tgt, const void *save)
777
1571
char *old= *(char **) tgt;
778
1572
*(char **)tgt= *(char **) save;
779
1573
if (var->flags & PLUGIN_VAR_MEMALLOC)
781
*(char **)tgt= strdup(*(char **) save);
784
* There isn't a _really_ good thing to do here until this whole set_var
785
* mess gets redesigned
788
errmsg_printf(ERRMSG_LVL_ERROR, _("Out of memory."));
1575
*(char **)tgt= my_strdup(*(char **) save, MYF(0));
1576
my_free(old, MYF(0));
829
static const string make_bookmark_name(const string &plugin, const char *name, int flags)
831
/* Embed the flags into the first char of the string */
832
string varname(1, static_cast<char>(flags & PLUGIN_VAR_TYPEMASK));
833
varname.append(plugin);
834
varname.push_back('_');
835
varname.append(name);
837
for (string::iterator p= varname.begin() + 1; p != varname.end(); ++p)
848
1622
called by register_var, construct_options and test_plugin_options.
849
1623
Returns the 'bookmark' for the named variable.
850
1624
LOCK_system_variables_hash should be at least read locked
852
static st_bookmark *find_bookmark(const string &plugin, const char *name, int flags)
1626
static st_bookmark *find_bookmark(const char *plugin, const char *name, int flags)
854
1628
st_bookmark *result= NULL;
1629
uint namelen, length, pluginlen= 0;
856
if (!(flags & PLUGIN_VAR_SessionLOCAL))
1632
if (!(flags & PLUGIN_VAR_THDLOCAL))
859
const string varname(make_bookmark_name(plugin, name, flags));
1635
namelen= strlen(name);
1637
pluginlen= strlen(plugin) + 1;
1638
length= namelen + pluginlen + 2;
1639
varname= (char*) my_alloca(length);
1643
strxmov(varname + 1, plugin, "_", name, NullS);
1644
for (p= varname + 1; *p; p++)
1649
memcpy(varname + 1, name, namelen + 1);
1651
varname[0]= flags & PLUGIN_VAR_TYPEMASK;
862
1653
result= (st_bookmark*) hash_search(&bookmark_hash,
863
(const unsigned char*) varname.c_str(), varname.size() - 1);
1654
(const uchar*) varname, length - 1);
870
returns a bookmark for session-local variables, creating if neccessary.
871
returns null for non session-local variables.
1662
returns a bookmark for thd-local variables, creating if neccessary.
1663
returns null for non thd-local variables.
872
1664
Requires that a write lock is obtained on LOCK_system_variables_hash
874
static st_bookmark *register_var(const string &plugin, const char *name,
1666
static st_bookmark *register_var(const char *plugin, const char *name,
877
if (!(flags & PLUGIN_VAR_SessionLOCAL))
880
uint32_t size= 0, offset, new_size;
1669
uint length= strlen(plugin) + strlen(name) + 3, size= 0, offset, new_size;
881
1670
st_bookmark *result;
1673
if (!(flags & PLUGIN_VAR_THDLOCAL))
883
1676
switch (flags & PLUGIN_VAR_TYPEMASK) {
884
1677
case PLUGIN_VAR_BOOL:
885
size= ALIGN_SIZE(sizeof(bool));
1678
size= sizeof(my_bool);
887
1680
case PLUGIN_VAR_INT:
888
size= ALIGN_SIZE(sizeof(int));
890
1683
case PLUGIN_VAR_LONG:
891
size= ALIGN_SIZE(sizeof(long));
1684
case PLUGIN_VAR_ENUM:
893
1687
case PLUGIN_VAR_LONGLONG:
894
size= ALIGN_SIZE(sizeof(uint64_t));
1688
case PLUGIN_VAR_SET:
1689
size= sizeof(uint64_t);
896
1691
case PLUGIN_VAR_STR:
897
size= ALIGN_SIZE(sizeof(char*));
1692
size= sizeof(char*);
1699
varname= ((char*) my_alloca(length));
1700
strxmov(varname + 1, plugin, "_", name, NullS);
1701
for (p= varname + 1; *p; p++)
905
if (!(result= find_bookmark(plugin, name, flags)))
1705
if (!(result= find_bookmark(NULL, varname + 1, flags)))
907
const string varname(make_bookmark_name(plugin, name, flags));
909
result= static_cast<st_bookmark*>(alloc_root(&plugin_mem_root,
910
sizeof(struct st_bookmark) + varname.size() + 1));
911
memset(result->key, 0, varname.size()+1);
912
memcpy(result->key, varname.c_str(), varname.size());
913
result->name_len= varname.size() - 2;
1707
result= (st_bookmark*) alloc_root(&plugin_mem_root,
1708
sizeof(struct st_bookmark) + length-1);
1709
varname[0]= flags & PLUGIN_VAR_TYPEMASK;
1710
memcpy(result->key, varname, length);
1711
result->name_len= length - 2;
914
1712
result->offset= -1;
916
1714
assert(size && !(size & (size-1))); /* must be power of 2 */
1033
1824
if ((pi->plugin_var->flags & PLUGIN_VAR_TYPEMASK) == PLUGIN_VAR_STR &&
1034
1825
pi->plugin_var->flags & PLUGIN_VAR_MEMALLOC)
1036
char **pp= (char**) (session->variables.dynamic_variables_ptr +
1827
char **pp= (char**) (thd->variables.dynamic_variables_ptr +
1037
1828
*(int*)(pi->plugin_var + 1));
1038
1829
if ((*pp= *(char**) (global_system_variables.dynamic_variables_ptr +
1039
1830
*(int*)(pi->plugin_var + 1))))
1831
*pp= my_strdup(*pp, MYF(MY_WME|MY_FAE));
1046
1835
if (global_lock)
1047
1836
pthread_mutex_unlock(&LOCK_global_system_variables);
1049
session->variables.dynamic_variables_version=
1838
thd->variables.dynamic_variables_version=
1050
1839
global_system_variables.dynamic_variables_version;
1051
session->variables.dynamic_variables_head=
1840
thd->variables.dynamic_variables_head=
1052
1841
global_system_variables.dynamic_variables_head;
1053
session->variables.dynamic_variables_size=
1842
thd->variables.dynamic_variables_size=
1054
1843
global_system_variables.dynamic_variables_size;
1056
pthread_rwlock_unlock(&LOCK_system_variables_hash);
1845
rw_unlock(&LOCK_system_variables_hash);
1058
return (unsigned char*)session->variables.dynamic_variables_ptr + offset;
1061
static bool *mysql_sys_var_ptr_bool(Session* a_session, int offset)
1063
return (bool *)intern_sys_var_ptr(a_session, offset, true);
1066
static int *mysql_sys_var_ptr_int(Session* a_session, int offset)
1068
return (int *)intern_sys_var_ptr(a_session, offset, true);
1071
static long *mysql_sys_var_ptr_long(Session* a_session, int offset)
1073
return (long *)intern_sys_var_ptr(a_session, offset, true);
1076
static int64_t *mysql_sys_var_ptr_int64_t(Session* a_session, int offset)
1078
return (int64_t *)intern_sys_var_ptr(a_session, offset, true);
1081
static char **mysql_sys_var_ptr_str(Session* a_session, int offset)
1083
return (char **)intern_sys_var_ptr(a_session, offset, true);
1086
void plugin_sessionvar_init(Session *session)
1088
session->variables.storage_engine= NULL;
1089
cleanup_variables(session, &session->variables);
1091
session->variables= global_system_variables;
1092
session->variables.storage_engine= NULL;
1847
return (uchar*)thd->variables.dynamic_variables_ptr + offset;
1850
static bool *mysql_sys_var_ptr_bool(THD* a_thd, int offset)
1852
return (bool *)intern_sys_var_ptr(a_thd, offset, true);
1855
static int *mysql_sys_var_ptr_int(THD* a_thd, int offset)
1857
return (int *)intern_sys_var_ptr(a_thd, offset, true);
1860
static long *mysql_sys_var_ptr_long(THD* a_thd, int offset)
1862
return (long *)intern_sys_var_ptr(a_thd, offset, true);
1865
static int64_t *mysql_sys_var_ptr_int64_t(THD* a_thd, int offset)
1867
return (int64_t *)intern_sys_var_ptr(a_thd, offset, true);
1870
static char **mysql_sys_var_ptr_str(THD* a_thd, int offset)
1872
return (char **)intern_sys_var_ptr(a_thd, offset, true);
1875
static uint64_t *mysql_sys_var_ptr_set(THD* a_thd, int offset)
1877
return (uint64_t *)intern_sys_var_ptr(a_thd, offset, true);
1880
static unsigned long *mysql_sys_var_ptr_enum(THD* a_thd, int offset)
1882
return (unsigned long *)intern_sys_var_ptr(a_thd, offset, true);
1886
void plugin_thdvar_init(THD *thd)
1888
plugin_ref old_table_plugin= thd->variables.table_plugin;
1890
thd->variables.table_plugin= NULL;
1891
cleanup_variables(thd, &thd->variables);
1893
thd->variables= global_system_variables;
1894
thd->variables.table_plugin= NULL;
1094
1896
/* we are going to allocate these lazily */
1095
session->variables.dynamic_variables_version= 0;
1096
session->variables.dynamic_variables_size= 0;
1097
session->variables.dynamic_variables_ptr= 0;
1897
thd->variables.dynamic_variables_version= 0;
1898
thd->variables.dynamic_variables_size= 0;
1899
thd->variables.dynamic_variables_ptr= 0;
1099
session->variables.storage_engine= global_system_variables.storage_engine;
1901
thd->variables.table_plugin=
1902
my_intern_plugin_lock(NULL, global_system_variables.table_plugin);
1903
intern_plugin_unlock(NULL, old_table_plugin);
1104
1909
Unlocks all system variables which hold a reference
1106
static void unlock_variables(Session *, struct system_variables *vars)
1911
static void unlock_variables(THD *thd __attribute__((__unused__)),
1912
struct system_variables *vars)
1108
vars->storage_engine= NULL;
1914
intern_plugin_unlock(NULL, vars->table_plugin);
1915
vars->table_plugin= NULL;
1294
2148
pthread_mutex_lock(&LOCK_global_system_variables);
1295
tgt= real_value_ptr(session, type);
2149
tgt= real_value_ptr(thd, type);
1296
2150
src= ((void **) (plugin_var + 1) + 1);
1298
if (plugin_var->flags & PLUGIN_VAR_SessionLOCAL)
2152
if (plugin_var->flags & PLUGIN_VAR_THDLOCAL)
1300
2154
if (type != OPT_GLOBAL)
1301
src= real_value_ptr(session, OPT_GLOBAL);
2155
src= real_value_ptr(thd, OPT_GLOBAL);
1303
2157
switch (plugin_var->flags & PLUGIN_VAR_TYPEMASK) {
1304
2158
case PLUGIN_VAR_INT:
1305
src= &((sessionvar_uint_t*) plugin_var)->def_val;
2159
src= &((thdvar_uint_t*) plugin_var)->def_val;
1307
2161
case PLUGIN_VAR_LONG:
1308
src= &((sessionvar_ulong_t*) plugin_var)->def_val;
2162
src= &((thdvar_ulong_t*) plugin_var)->def_val;
1310
2164
case PLUGIN_VAR_LONGLONG:
1311
src= &((sessionvar_uint64_t_t*) plugin_var)->def_val;
2165
src= &((thdvar_uint64_t_t*) plugin_var)->def_val;
2167
case PLUGIN_VAR_ENUM:
2168
src= &((thdvar_enum_t*) plugin_var)->def_val;
2170
case PLUGIN_VAR_SET:
2171
src= &((thdvar_set_t*) plugin_var)->def_val;
1313
2173
case PLUGIN_VAR_BOOL:
1314
src= &((sessionvar_bool_t*) plugin_var)->def_val;
2174
src= &((thdvar_bool_t*) plugin_var)->def_val;
1316
2176
case PLUGIN_VAR_STR:
1317
src= &((sessionvar_str_t*) plugin_var)->def_val;
2177
src= &((thdvar_str_t*) plugin_var)->def_val;
1324
/* session must equal current_session if PLUGIN_VAR_SessionLOCAL flag is set */
1325
assert(!(plugin_var->flags & PLUGIN_VAR_SessionLOCAL) ||
1326
session == current_session);
2184
/* thd must equal current_thd if PLUGIN_VAR_THDLOCAL flag is set */
2185
assert(!(plugin_var->flags & PLUGIN_VAR_THDLOCAL) ||
2186
thd == current_thd);
1328
if (!(plugin_var->flags & PLUGIN_VAR_SessionLOCAL) || type == OPT_GLOBAL)
2188
if (!(plugin_var->flags & PLUGIN_VAR_THDLOCAL) || type == OPT_GLOBAL)
1330
plugin_var->update(session, plugin_var, tgt, src);
2190
plugin_var->update(thd, plugin_var, tgt, src);
1331
2191
pthread_mutex_unlock(&LOCK_global_system_variables);
1335
2195
pthread_mutex_unlock(&LOCK_global_system_variables);
1336
plugin_var->update(session, plugin_var, tgt, src);
2196
plugin_var->update(thd, plugin_var, tgt, src);
1341
bool sys_var_pluginvar::update(Session *session, set_var *var)
2201
bool sys_var_pluginvar::update(THD *thd, set_var *var)
1345
2205
assert(is_readonly() || plugin_var->update);
1347
/* session must equal current_session if PLUGIN_VAR_SessionLOCAL flag is set */
1348
assert(!(plugin_var->flags & PLUGIN_VAR_SessionLOCAL) ||
1349
session == current_session);
2207
/* thd must equal current_thd if PLUGIN_VAR_THDLOCAL flag is set */
2208
assert(!(plugin_var->flags & PLUGIN_VAR_THDLOCAL) ||
2209
thd == current_thd);
1351
2211
if (is_readonly())
1354
2214
pthread_mutex_lock(&LOCK_global_system_variables);
1355
tgt= real_value_ptr(session, var->type);
2215
tgt= real_value_ptr(thd, var->type);
1357
if (!(plugin_var->flags & PLUGIN_VAR_SessionLOCAL) || var->type == OPT_GLOBAL)
2217
if (!(plugin_var->flags & PLUGIN_VAR_THDLOCAL) || var->type == OPT_GLOBAL)
1359
2219
/* variable we are updating has global scope, so we unlock after updating */
1360
plugin_var->update(session, plugin_var, tgt, &var->save_result);
2220
plugin_var->update(thd, plugin_var, tgt, &var->save_result);
1361
2221
pthread_mutex_unlock(&LOCK_global_system_variables);
1365
2225
pthread_mutex_unlock(&LOCK_global_system_variables);
1366
plugin_var->update(session, plugin_var, tgt, &var->save_result);
2226
plugin_var->update(thd, plugin_var, tgt, &var->save_result);
1372
2232
#define OPTION_SET_LIMITS(type, options, opt) \
1373
options->var_type= type; \
1374
options->def_value= (opt)->def_val; \
1375
options->min_value= (opt)->min_val; \
1376
options->max_value= (opt)->max_val; \
2233
options->var_type= type; \
2234
options->def_value= (opt)->def_val; \
2235
options->min_value= (opt)->min_val; \
2236
options->max_value= (opt)->max_val; \
1377
2237
options->block_size= (long) (opt)->blk_sz
1380
void plugin_opt_set_limits(struct my_option *options,
1381
const drizzle_sys_var *opt)
2240
static void plugin_opt_set_limits(struct my_option *options,
2241
const struct st_mysql_sys_var *opt)
1383
2243
options->sub_size= 0;
1385
2245
switch (opt->flags & (PLUGIN_VAR_TYPEMASK |
1386
PLUGIN_VAR_UNSIGNED | PLUGIN_VAR_SessionLOCAL)) {
2246
PLUGIN_VAR_UNSIGNED | PLUGIN_VAR_THDLOCAL)) {
1387
2247
/* global system variables */
1388
2248
case PLUGIN_VAR_INT:
1389
2249
OPTION_SET_LIMITS(GET_INT, options, (sysvar_int_t*) opt);
1450
2338
options->arg_type= OPT_ARG;
1453
static bool get_one_plugin_option(int, const struct my_option *, char *)
2341
extern "C" bool get_one_plugin_option(int optid, const struct my_option *,
2344
bool get_one_plugin_option(int optid __attribute__((unused)),
2345
const struct my_option *opt __attribute__((__unused__)),
2346
char *argument __attribute__((__unused__)))
1459
static int construct_options(memory::Root *mem_root, plugin::Module *tmp,
2352
static int construct_options(MEM_ROOT *mem_root, struct st_plugin_int *tmp,
2353
my_option *options, bool can_disable)
1463
int localoptionid= 256;
1464
const string plugin_name(tmp->getManifest().name);
1466
size_t namelen= plugin_name.size(), optnamelen;
2355
const char *plugin_name= tmp->plugin->name;
2356
uint namelen= strlen(plugin_name), optnamelen;
2357
uint buffer_length= namelen * 4 + (can_disable ? 75 : 10);
2358
char *name= (char*) alloc_root(mem_root, buffer_length) + 1;
1468
2359
char *optname, *p;
1469
2360
int index= 0, offset= 0;
1470
drizzle_sys_var *opt, **plugin_option;
2361
st_mysql_sys_var *opt, **plugin_option;
1471
2362
st_bookmark *v;
1473
string name(plugin_name);
1474
transform(name.begin(), name.end(), name.begin(), ::tolower);
1476
for (string::iterator iter= name.begin(); iter != name.end(); ++iter)
2364
/* support --skip-plugin-foo syntax */
2365
memcpy(name, plugin_name, namelen + 1);
2366
my_casedn_str(&my_charset_latin1, name);
2367
strxmov(name + namelen + 1, "plugin-", name, NullS);
2368
/* Now we have namelen + 1 + 7 + namelen + 1 == namelen * 2 + 9. */
2370
for (p= name + namelen*2 + 8; p > name; p--)
2376
strxmov(name + namelen*2 + 10, "Enable ", plugin_name, " plugin. "
2377
"Disable with --skip-", name," (will save memory).", NullS);
2379
Now we have namelen * 2 + 10 (one char unused) + 7 + namelen + 9 +
2380
20 + namelen + 20 + 1 == namelen * 4 + 67.
2383
options[0].comment= name + namelen*2 + 10;
2386
options[1].name= (options[0].name= name) + namelen + 1;
2387
options[0].id= options[1].id= 256; /* must be >255. dup id ok */
2388
options[0].var_type= options[1].var_type= GET_BOOL;
2389
options[0].arg_type= options[1].arg_type= NO_ARG;
2390
options[0].def_value= options[1].def_value= true;
2391
options[0].value= options[0].u_max_value=
2392
options[1].value= options[1].u_max_value= (char**) (name - 1);
1483
2396
Two passes as the 2nd pass will take pointer addresses for use
1484
2397
by my_getopt and register_var() in the first pass uses realloc
1487
for (plugin_option= tmp->getManifest().system_vars;
2400
for (plugin_option= tmp->plugin->system_vars;
1488
2401
plugin_option && *plugin_option; plugin_option++, index++)
1490
2403
opt= *plugin_option;
1491
if (!(opt->flags & PLUGIN_VAR_SessionLOCAL))
2404
if (!(opt->flags & PLUGIN_VAR_THDLOCAL))
1493
2406
if (!(register_var(name, opt->name, opt->flags)))
1495
2408
switch (opt->flags & PLUGIN_VAR_TYPEMASK) {
1496
2409
case PLUGIN_VAR_BOOL:
1497
(((sessionvar_bool_t *)opt)->resolve)= mysql_sys_var_ptr_bool;
2410
(((thdvar_bool_t *)opt)->resolve)= mysql_sys_var_ptr_bool;
1499
2412
case PLUGIN_VAR_INT:
1500
(((sessionvar_int_t *)opt)->resolve)= mysql_sys_var_ptr_int;
2413
(((thdvar_int_t *)opt)->resolve)= mysql_sys_var_ptr_int;
1502
2415
case PLUGIN_VAR_LONG:
1503
(((sessionvar_long_t *)opt)->resolve)= mysql_sys_var_ptr_long;
2416
(((thdvar_long_t *)opt)->resolve)= mysql_sys_var_ptr_long;
1505
2418
case PLUGIN_VAR_LONGLONG:
1506
(((sessionvar_int64_t_t *)opt)->resolve)= mysql_sys_var_ptr_int64_t;
2419
(((thdvar_int64_t_t *)opt)->resolve)= mysql_sys_var_ptr_int64_t;
1508
2421
case PLUGIN_VAR_STR:
1509
(((sessionvar_str_t *)opt)->resolve)= mysql_sys_var_ptr_str;
2422
(((thdvar_str_t *)opt)->resolve)= mysql_sys_var_ptr_str;
2424
case PLUGIN_VAR_ENUM:
2425
(((thdvar_enum_t *)opt)->resolve)= mysql_sys_var_ptr_enum;
2427
case PLUGIN_VAR_SET:
2428
(((thdvar_set_t *)opt)->resolve)= mysql_sys_var_ptr_set;
1512
errmsg_printf(ERRMSG_LVL_ERROR, _("Unknown variable type code 0x%x in plugin '%s'."),
1513
opt->flags, plugin_name.c_str());
2431
sql_print_error("Unknown variable type code 0x%x in plugin '%s'.",
2432
opt->flags, plugin_name);
1518
for (plugin_option= tmp->getManifest().system_vars;
2437
for (plugin_option= tmp->plugin->system_vars;
1519
2438
plugin_option && *plugin_option; plugin_option++, index++)
1521
2440
switch ((opt= *plugin_option)->flags & PLUGIN_VAR_TYPEMASK) {