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 */
16
#include <drizzled/server_includes.h>
17
#include <mysys/my_getopt.h>
18
#include <mysys/hash.h>
20
#include <drizzled/authentication.h>
21
#include <drizzled/logging.h>
22
#include <drizzled/errmsg.h>
23
#include <drizzled/qcache.h>
24
#include <drizzled/sql_parse.h>
25
#include <drizzled/scheduling.h>
26
#include <drizzled/replication_services.h>
27
#include <drizzled/show.h>
28
#include <drizzled/handler.h>
29
#include <drizzled/set_var.h>
30
#include <drizzled/session.h>
31
#include <drizzled/item/null.h>
32
#include <drizzled/plugin_registry.h>
23
37
#include <algorithm>
25
#include "drizzled/option.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) */
39
#include <drizzled/error.h>
40
#include <drizzled/gettext.h>
42
#define REPORT_TO_LOG 1
43
#define REPORT_TO_USER 2
51
45
using namespace std;
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
46
using namespace drizzled;
48
typedef plugin::Manifest builtin_plugin[];
49
extern builtin_plugin DRIZZLED_BUILTIN_LIST;
50
static plugin::Manifest *drizzled_builtins[]=
52
DRIZZLED_BUILTIN_LIST,(plugin::Manifest *)NULL
64
54
class sys_var_pluginvar;
65
55
static vector<sys_var_pluginvar *> plugin_sysvar_vec;
67
char *opt_plugin_add= NULL;
68
char *opt_plugin_remove= NULL;
69
57
char *opt_plugin_load= NULL;
58
const char *opt_plugin_load_default= QUOTE_ARG(DRIZZLED_PLUGIN_LIST);
70
59
char *opt_plugin_dir_ptr;
71
60
char opt_plugin_dir[FN_REFLEN];
72
const char *opt_plugin_load_default= PANDORA_PLUGIN_LIST;
61
static const char *plugin_declarations_sym= "_drizzled_plugin_declaration_";
74
63
/* Note that 'int version' must be the first field of every plugin
75
64
sub-structure (plugin->info).
114
skeleton of a plugin variable - portion of structure common to all.
116
struct st_mysql_sys_var
118
DRIZZLE_PLUGIN_VAR_HEADER;
123
123
sys_var class for access to all plugin variables visible to the user
125
125
class sys_var_pluginvar: public sys_var
128
plugin::Module *plugin;
129
drizzle_sys_var *plugin_var;
128
plugin::Handle *plugin;
129
struct st_mysql_sys_var *plugin_var;
131
131
sys_var_pluginvar(const std::string name_arg,
132
drizzle_sys_var *plugin_var_arg)
132
struct st_mysql_sys_var *plugin_var_arg)
133
133
:sys_var(name_arg), plugin_var(plugin_var_arg) {}
134
134
sys_var_pluginvar *cast_pluginvar() { return this; }
135
135
bool is_readonly() const { return plugin_var->flags & PLUGIN_VAR_READONLY; }
136
bool check_type(sql_var_t type)
136
bool check_type(enum_var_type type)
137
137
{ return !(plugin_var->flags & PLUGIN_VAR_SessionLOCAL) && type != OPT_GLOBAL; }
138
138
bool check_update_type(Item_result type);
139
139
SHOW_TYPE show_type();
140
unsigned char* real_value_ptr(Session *session, sql_var_t type);
140
unsigned char* real_value_ptr(Session *session, enum_var_type type);
141
141
TYPELIB* plugin_var_typelib(void);
142
unsigned char* value_ptr(Session *session, sql_var_t type,
142
unsigned char* value_ptr(Session *session, enum_var_type type,
143
143
const LEX_STRING *base);
144
144
bool check(Session *session, set_var *var);
145
bool check_default(sql_var_t)
145
bool check_default(enum_var_type)
146
146
{ return is_readonly(); }
147
void set_default(Session *session, sql_var_t);
147
void set_default(Session *session, enum_var_type);
148
148
bool update(Session *session, 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 set<string> &plugin_list);
158
static int test_plugin_options(memory::Root *, plugin::Module *,
153
static bool plugin_load_list(MEM_ROOT *tmp_root, int *argc, char **argv,
155
static int test_plugin_options(MEM_ROOT *, plugin::Handle *,
157
static bool register_builtin(plugin::Handle *,
160
159
static void unlock_variables(Session *session, struct system_variables *vars);
161
160
static void cleanup_variables(Session *session, struct system_variables *vars);
162
161
static void plugin_vars_free_values(sys_var *vars);
162
static void plugin_opt_set_limits(struct my_option *options,
163
const struct st_mysql_sys_var *opt);
164
static void reap_plugins(void);
164
167
/* declared in set_var.cc */
165
168
extern sys_var *intern_find_sys_var(const char *str, uint32_t length, bool no_error);
231
234
Plugin support code
232
235
****************************************************************************/
237
static plugin::Library *plugin_dl_find(const LEX_STRING *dl)
240
plugin::Library *tmp;
242
for (i= 0; i < plugin_dl_array.elements; i++)
244
tmp= *dynamic_element(&plugin_dl_array, i, plugin::Library **);
245
if (! my_strnncoll(files_charset_info,
246
(const unsigned char *)dl->str, dl->length,
247
(const unsigned char *)tmp->dl.str, tmp->dl.length))
253
static plugin::Library *plugin_dl_insert_or_reuse(plugin::Library *plugin_dl)
256
plugin::Library *tmp;
258
for (i= 0; i < plugin_dl_array.elements; i++)
260
tmp= *dynamic_element(&plugin_dl_array, i, plugin::Library **);
262
memcpy(tmp, plugin_dl, sizeof(plugin::Library));
266
if (insert_dynamic(&plugin_dl_array, (unsigned char*)&plugin_dl))
268
tmp= *dynamic_element(&plugin_dl_array, plugin_dl_array.elements - 1,
270
(plugin::Library *) memdup_root(&plugin_mem_root,
271
(unsigned char*)plugin_dl,
272
sizeof(plugin::Library));
276
static inline void free_plugin_mem(plugin::Library *p)
284
static plugin::Library *plugin_dl_add(const LEX_STRING *dl, int report)
287
uint32_t plugin_dir_len;
288
plugin::Library *tmp, plugin_dl;
290
plugin_dir_len= strlen(opt_plugin_dir);
291
dlpath.reserve(FN_REFLEN);
293
Ensure that the dll doesn't have a path.
294
This is done to ensure that only approved libraries from the
295
plugin directory are used (to make this even remotely secure).
297
if (strchr(dl->str, FN_LIBCHAR) ||
298
check_string_char_length((LEX_STRING *) dl, "", NAME_CHAR_LEN,
299
system_charset_info, 1) ||
300
plugin_dir_len + dl->length + 1 >= FN_REFLEN)
302
if (report & REPORT_TO_USER)
303
my_error(ER_UDF_NO_PATHS, MYF(0));
304
if (report & REPORT_TO_LOG)
305
errmsg_printf(ERRMSG_LVL_ERROR, "%s",ER(ER_UDF_NO_PATHS));
308
/* If this dll is already loaded just increase ref_count. */
309
if ((tmp= plugin_dl_find(dl)))
313
memset(&plugin_dl, 0, sizeof(plugin_dl));
314
/* Compile dll path */
315
dlpath.append(opt_plugin_dir);
317
dlpath.append(dl->str);
318
/* Open new dll handle */
319
if (!(plugin_dl.handle= dlopen(dlpath.c_str(), RTLD_LAZY|RTLD_GLOBAL)))
321
const char *errmsg=dlerror();
322
uint32_t dlpathlen= dlpath.length();
323
if (!dlpath.compare(0, dlpathlen, errmsg))
324
{ // if errmsg starts from dlpath, trim this prefix.
326
if (*errmsg == ':') errmsg++;
327
if (*errmsg == ' ') errmsg++;
329
if (report & REPORT_TO_USER)
330
my_error(ER_CANT_OPEN_LIBRARY, MYF(0), dlpath.c_str(), errno, errmsg);
331
if (report & REPORT_TO_LOG)
332
errmsg_printf(ERRMSG_LVL_ERROR, ER(ER_CANT_OPEN_LIBRARY), dlpath.c_str(), errno, errmsg);
336
/* Find plugin declarations */
337
if (!(sym= dlsym(plugin_dl.handle, plugin_declarations_sym)))
339
free_plugin_mem(&plugin_dl);
340
if (report & REPORT_TO_USER)
341
my_error(ER_CANT_FIND_DL_ENTRY, MYF(0), plugin_declarations_sym);
342
if (report & REPORT_TO_LOG)
343
errmsg_printf(ERRMSG_LVL_ERROR, ER(ER_CANT_FIND_DL_ENTRY), plugin_declarations_sym);
347
plugin_dl.plugins= static_cast<plugin::Manifest *>(sym);
349
/* Duplicate and convert dll name */
350
plugin_dl.dl.length= dl->length * files_charset_info->mbmaxlen + 1;
351
if (! (plugin_dl.dl.str= (char*) calloc(plugin_dl.dl.length, sizeof(char))))
353
free_plugin_mem(&plugin_dl);
354
if (report & REPORT_TO_USER)
355
my_error(ER_OUTOFMEMORY, MYF(0), plugin_dl.dl.length);
356
if (report & REPORT_TO_LOG)
357
errmsg_printf(ERRMSG_LVL_ERROR, ER(ER_OUTOFMEMORY), plugin_dl.dl.length);
360
strcpy(plugin_dl.dl.str, dl->str);
361
/* Add this dll to array */
362
if (! (tmp= plugin_dl_insert_or_reuse(&plugin_dl)))
364
free_plugin_mem(&plugin_dl);
365
if (report & REPORT_TO_USER)
366
my_error(ER_OUTOFMEMORY, MYF(0), sizeof(plugin::Library));
367
if (report & REPORT_TO_LOG)
368
errmsg_printf(ERRMSG_LVL_ERROR, ER(ER_OUTOFMEMORY),
369
sizeof(plugin::Library));
376
static void plugin_dl_del(const LEX_STRING *dl)
380
for (i= 0; i < plugin_dl_array.elements; i++)
382
plugin::Library *tmp= *dynamic_element(&plugin_dl_array, i,
384
if (! my_strnncoll(files_charset_info,
385
(const unsigned char *)dl->str, dl->length,
386
(const unsigned char *)tmp->dl.str, tmp->dl.length))
388
/* Do not remove this element, unless no other plugin uses this dll. */
390
free_plugin_mem(tmp);
391
memset(tmp, 0, sizeof(plugin::Library));
401
static plugin::Handle *plugin_insert_or_reuse(plugin::Handle *plugin)
403
if (insert_dynamic(&plugin_array, (unsigned char*)&plugin))
405
plugin= *dynamic_element(&plugin_array, plugin_array.elements - 1,
239
413
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)
415
static bool plugin_add(MEM_ROOT *tmp_root,
416
const LEX_STRING *name, const LEX_STRING *dl,
417
int *argc, char **argv, int report)
419
PluginRegistry ®istry= PluginRegistry::getPluginRegistry();
421
plugin::Manifest *manifest;
245
422
if (! initialized)
248
if (registry.find(library->getName()))
425
if (registry.find(name))
250
errmsg_printf(ERRMSG_LVL_WARN, ER(ER_PLUGIN_EXISTS),
251
library->getName().c_str());
427
if (report & REPORT_TO_USER)
428
my_error(ER_UDF_EXISTS, MYF(0), name->str);
429
if (report & REPORT_TO_LOG)
430
errmsg_printf(ERRMSG_LVL_ERROR, ER(ER_UDF_EXISTS), name->str);
433
plugin::Library *library= plugin_dl_add(dl, report);
255
plugin::Module *tmp= NULL;
437
plugin::Handle *tmp= NULL;
256
438
/* 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());
439
for (manifest= library->plugins; manifest->name; manifest++)
441
if (! my_strnncoll(system_charset_info,
442
(const unsigned char *)name->str, name->length,
443
(const unsigned char *)manifest->name,
444
strlen(manifest->name)))
446
tmp= new (std::nothrow) plugin::Handle(manifest, library);
450
if (!test_plugin_options(tmp_root, tmp, argc, argv))
452
if ((tmp= plugin_insert_or_reuse(tmp)))
455
init_alloc_root(&tmp->mem_root, 4096, 4096);
458
mysql_del_sys_var_chain(tmp->system_vars);
461
/* plugin was disabled */
466
if (report & REPORT_TO_USER)
467
my_error(ER_CANT_FIND_DL_ENTRY, MYF(0), name->str);
468
if (report & REPORT_TO_LOG)
469
errmsg_printf(ERRMSG_LVL_ERROR, ER(ER_CANT_FIND_DL_ENTRY), name->str);
284
static void delete_module(plugin::Module *module)
476
static void plugin_del(plugin::Handle *plugin)
478
PluginRegistry ®istry= PluginRegistry::getPluginRegistry();
479
if (plugin->isInited)
481
if (plugin->getManifest().status_vars)
483
remove_status_vars(plugin->getManifest().status_vars);
486
if (plugin->getManifest().deinit)
487
plugin->getManifest().deinit(registry);
286
490
/* Free allocated strings before deleting the plugin. */
287
plugin_vars_free_values(module->system_vars);
288
module->isInited= false;
491
plugin_vars_free_values(plugin->system_vars);
492
if (plugin->plugin_dl)
493
plugin_dl_del(&plugin->plugin_dl->dl);
494
plugin->isInited= false;
289
495
pthread_rwlock_wrlock(&LOCK_system_variables_hash);
290
mysql_del_sys_var_chain(module->system_vars);
496
mysql_del_sys_var_chain(plugin->system_vars);
291
497
pthread_rwlock_unlock(&LOCK_system_variables_hash);
296
static void reap_plugins(plugin::Registry ®istry)
501
static void reap_plugins(void)
298
std::map<std::string, plugin::Module *>::const_iterator modules=
299
registry.getModulesMap().begin();
301
while (modules != registry.getModulesMap().end())
505
drizzled::plugin::Handle *plugin;
507
count= plugin_array.elements;
509
for (idx= 0; idx < count; idx++)
303
plugin::Module *module= (*modules).second;
304
delete_module(module);
511
plugin= *dynamic_element(&plugin_array, idx, drizzled::plugin::Handle **);
308
514
drizzle_del_plugin_sysvar();
312
static void plugin_initialize_vars(plugin::Module *module)
517
static bool plugin_initialize(drizzled::plugin::Handle *plugin)
519
assert(plugin->isInited == false);
521
PluginRegistry ®istry= PluginRegistry::getPluginRegistry();
522
if (plugin->getManifest().init)
524
if (plugin->getManifest().init(registry))
526
errmsg_printf(ERRMSG_LVL_ERROR,
527
_("Plugin '%s' init function returned error.\n"),
528
plugin->getName().c_str());
532
plugin->isInited= true;
534
if (plugin->getManifest().status_vars)
536
add_status_vars(plugin->getManifest().status_vars); // add_status_vars makes a copy
315
540
set the plugin attribute of plugin's sys vars so they are pointing
316
541
to the active plugin
318
if (module->system_vars)
543
if (plugin->system_vars)
320
sys_var_pluginvar *var= module->system_vars->cast_pluginvar();
545
sys_var_pluginvar *var= plugin->system_vars->cast_pluginvar();
324
549
if (! var->getNext())
326
551
var= var->getNext()->cast_pluginvar();
332
static bool plugin_initialize(plugin::Registry ®istry,
333
plugin::Module *module)
335
assert(module->isInited == false);
337
plugin::Context loading_context(registry, module);
338
if (module->getManifest().init)
340
if (module->getManifest().init(loading_context))
342
errmsg_printf(ERRMSG_LVL_ERROR,
343
_("Plugin '%s' init function returned error.\n"),
344
module->getName().c_str());
348
module->isInited= true;
355
static unsigned char *get_bookmark_hash_key(const unsigned char *buff,
356
size_t *length, bool)
561
extern "C" unsigned char *get_bookmark_hash_key(const unsigned char *, size_t *, bool);
564
unsigned char *get_bookmark_hash_key(const unsigned char *buff, size_t *length, bool)
358
566
struct st_bookmark *var= (st_bookmark *)buff;
359
567
*length= var->name_len + 1;
399
609
for (builtins= drizzled_builtins; *builtins; builtins++)
402
if (manifest->name != NULL)
611
for (manifest= *builtins; manifest->name; manifest++)
404
module= new (std::nothrow) plugin::Module(manifest);
613
handle= new (std::nothrow) plugin::Handle(manifest);
408
free_root(&tmp_root, MYF(memory::MARK_BLOCKS_FREE));
409
if (test_plugin_options(&tmp_root, module, argc, argv))
617
free_root(&tmp_root, MYF(MY_MARK_BLOCKS_FREE));
618
if (test_plugin_options(&tmp_root, handle, argc, argv))
412
registry.add(module);
414
plugin_initialize_vars(module);
418
if (plugin_initialize(registry, module))
420
free_root(&tmp_root, MYF(0));
621
if (register_builtin(handle, &handle))
624
if (plugin_initialize(handle))
428
bool load_failed= false;
429
vector<string> plugin_list;
432
tokenize(opt_plugin_load, plugin_list, ",", true);
436
tokenize(opt_plugin_load_default, plugin_list, ",", true);
440
tokenize(opt_plugin_add, plugin_list, ",", true);
443
if (opt_plugin_remove)
445
vector<string> plugins_to_remove;
446
tokenize(opt_plugin_remove, plugins_to_remove, ",", true);
447
plugin_prune_list(plugin_list, plugins_to_remove);
450
/* Uniquify the list */
451
const set<string> plugin_list_set(plugin_list.begin(), plugin_list.end());
453
631
/* Register all dynamic plugins */
454
load_failed= plugin_load_list(registry, &tmp_root, argc, argv,
632
if (!(flags & PLUGIN_INIT_SKIP_DYNAMIC_LOADING))
458
free_root(&tmp_root, MYF(0));
635
plugin_load_list(&tmp_root, argc, argv, opt_plugin_load);
464
free_root(&tmp_root, MYF(0));
638
if (flags & PLUGIN_INIT_SKIP_INITIALIZATION)
469
642
Now we initialize all remaining plugins
471
std::map<std::string, plugin::Module *>::const_iterator modules=
472
registry.getModulesMap().begin();
474
while (modules != registry.getModulesMap().end())
644
for (idx= 0; idx < plugin_array.elements; idx++)
476
module= (*modules).second;
478
if (module->isInited == false)
646
handle= *dynamic_element(&plugin_array, idx, plugin::Handle **);
647
if (handle->isInited == false)
480
plugin_initialize_vars(module);
482
if (plugin_initialize(registry, module))
483
delete_module(module);
649
if (plugin_initialize(handle))
488
free_root(&tmp_root, MYF(0));
494
public unary_function<string, bool>
496
const string to_match;
498
PrunePlugin& operator=(const PrunePlugin&);
500
explicit PrunePlugin(const string &match_in) :
504
result_type operator()(const string &match_against)
506
return match_against == to_match;
510
static void plugin_prune_list(vector<string> &plugin_list,
511
const vector<string> &plugins_to_remove)
513
for (vector<string>::const_iterator iter= plugins_to_remove.begin();
514
iter != plugins_to_remove.end();
517
plugin_list.erase(remove_if(plugin_list.begin(),
656
free_root(&tmp_root, MYF(0));
662
free_root(&tmp_root, MYF(0));
667
static bool register_builtin(plugin::Handle *tmp,
668
plugin::Handle **ptr)
671
PluginRegistry ®istry= PluginRegistry::getPluginRegistry();
673
tmp->isInited= false;
676
if (insert_dynamic(&plugin_array, (unsigned char*)&tmp))
679
*ptr= *dynamic_element(&plugin_array, plugin_array.elements - 1,
525
689
called only by plugin_init()
527
static bool plugin_load_list(plugin::Registry ®istry,
528
memory::Root *tmp_root, int *argc, char **argv,
529
const set<string> &plugin_list)
691
static bool plugin_load_list(MEM_ROOT *tmp_root, int *argc, char **argv,
531
plugin::Library *library= NULL;
533
for (set<string>::const_iterator iter= plugin_list.begin();
534
iter != plugin_list.end();
694
char buffer[FN_REFLEN];
695
LEX_STRING name= {buffer, 0}, dl= {NULL, 0}, *str= &name;
696
plugin::Library *plugin_dl;
697
plugin::Manifest *plugin;
537
const string plugin_name(*iter);
538
library= registry.addLibrary(plugin_name);
701
if (p == buffer + sizeof(buffer) - 1)
541
errmsg_printf(ERRMSG_LVL_ERROR,
542
_("Couldn't load plugin library named '%s'.\n"),
543
plugin_name.c_str());
703
errmsg_printf(ERRMSG_LVL_ERROR, _("plugin-load parameter too long"));
547
free_root(tmp_root, MYF(memory::MARK_BLOCKS_FREE));
548
if (plugin_add(registry, tmp_root, library, argc, argv))
550
registry.removeLibrary(plugin_name);
551
errmsg_printf(ERRMSG_LVL_ERROR,
552
_("Couldn't load plugin named '%s'.\n"),
553
plugin_name.c_str());
707
switch ((*(p++)= *(list++))) {
709
list= NULL; /* terminate the loop */
711
case ':': /* can't use this as delimiter as it may be drive letter */
713
str->str[str->length]= '\0';
714
if (str == &name) // load all plugins in named module
718
p--; /* reset pointer */
723
if ((plugin_dl= plugin_dl_add(&dl, REPORT_TO_LOG)))
725
for (plugin= plugin_dl->plugins; plugin->name; plugin++)
727
name.str= (char *) plugin->name;
728
name.length= strlen(name.str);
730
free_root(tmp_root, MYF(MY_MARK_BLOCKS_FREE));
731
if (plugin_add(tmp_root, &name, &dl, argc, argv, REPORT_TO_LOG))
734
plugin_dl_del(&dl); // reduce ref count
739
free_root(tmp_root, MYF(MY_MARK_BLOCKS_FREE));
740
if (plugin_add(tmp_root, &name, &dl, argc, argv, REPORT_TO_LOG))
743
name.length= dl.length= 0;
744
dl.str= NULL; name.str= p= buffer;
751
name.str[name.length]= '\0';
763
errmsg_printf(ERRMSG_LVL_ERROR, _("Couldn't load plugin named '%s' with soname '%s'."),
562
void plugin_shutdown(plugin::Registry ®istry)
769
void plugin_shutdown(void)
772
size_t count= plugin_array.elements;
773
vector<plugin::Handle *> plugins;
774
vector<plugin::Library *> dl;
567
778
reap_needed= true;
569
reap_plugins(registry);
570
781
unlock_variables(NULL, &global_system_variables);
571
782
unlock_variables(NULL, &max_system_variables);
739
static void update_func_bool(Session *, drizzle_sys_var *,
963
static int check_func_enum(Session *, struct st_mysql_sys_var *var,
964
void *save, st_mysql_value *value)
966
char buff[STRING_BUFFER_USUAL_SIZE];
967
const char *strvalue= "NULL", *str;
973
if (var->flags & PLUGIN_VAR_SessionLOCAL)
974
typelib= ((sessionvar_enum_t*) var)->typelib;
976
typelib= ((sysvar_enum_t*) var)->typelib;
978
if (value->value_type(value) == DRIZZLE_VALUE_TYPE_STRING)
980
length= sizeof(buff);
981
if (!(str= value->val_str(value, buff, &length)))
983
if ((result= (long)find_type(typelib, str, length, 1)-1) < 0)
991
if (value->val_int(value, &tmp))
993
if (tmp >= typelib->count)
1001
*(long*)save= result;
1004
my_error(ER_WRONG_VALUE_FOR_VAR, MYF(0), var->name, strvalue);
1009
static int check_func_set(Session *, struct st_mysql_sys_var *var,
1010
void *save, st_mysql_value *value)
1012
char buff[STRING_BUFFER_USUAL_SIZE], *error= 0;
1013
const char *strvalue= "NULL", *str;
1020
if (var->flags & PLUGIN_VAR_SessionLOCAL)
1021
typelib= ((sessionvar_set_t*) var)->typelib;
1023
typelib= ((sysvar_set_t*)var)->typelib;
1025
if (value->value_type(value) == DRIZZLE_VALUE_TYPE_STRING)
1027
length= sizeof(buff);
1028
if (!(str= value->val_str(value, buff, &length)))
1030
result= find_set(typelib, str, length, NULL,
1031
&error, &error_len, ¬_used);
1034
length= min((uint32_t)sizeof(buff), error_len);
1035
strncpy(buff, error, length);
1043
if (value->val_int(value, (int64_t *)&result))
1045
if (unlikely((result >= (1UL << typelib->count)) &&
1046
(typelib->count < sizeof(long)*8)))
1048
llstr(result, buff);
1053
*(uint64_t*)save= result;
1056
my_error(ER_WRONG_VALUE_FOR_VAR, MYF(0), var->name, strvalue);
1061
static void update_func_bool(Session *, struct st_mysql_sys_var *,
740
1062
void *tgt, const void *save)
742
1064
*(bool *) tgt= *(int *) save ? 1 : 0;
746
static void update_func_int(Session *, drizzle_sys_var *,
1068
static void update_func_int(Session *, struct st_mysql_sys_var *,
747
1069
void *tgt, const void *save)
749
1071
*(int *)tgt= *(int *) save;
753
static void update_func_long(Session *, drizzle_sys_var *,
1075
static void update_func_long(Session *, struct st_mysql_sys_var *,
754
1076
void *tgt, const void *save)
756
1078
*(long *)tgt= *(long *) save;
760
static void update_func_int64_t(Session *, drizzle_sys_var *,
1082
static void update_func_int64_t(Session *, struct st_mysql_sys_var *,
761
1083
void *tgt, const void *save)
763
1085
*(int64_t *)tgt= *(uint64_t *) save;
767
static void update_func_str(Session *, drizzle_sys_var *var,
1089
static void update_func_str(Session *, struct st_mysql_sys_var *var,
768
1090
void *tgt, const void *save)
770
1092
char *old= *(char **) tgt;
1443
1850
options->arg_type= OPT_ARG;
1446
static int get_one_plugin_option(int, const struct option *, char *)
1853
extern "C" bool get_one_plugin_option(int optid, const struct my_option *,
1856
bool get_one_plugin_option(int, const struct my_option *, char *)
1452
static int construct_options(memory::Root *mem_root, plugin::Module *tmp,
1862
static int construct_options(MEM_ROOT *mem_root, plugin::Handle *tmp,
1863
my_option *options, bool can_disable)
1456
int localoptionid= 256;
1457
const string plugin_name(tmp->getManifest().name);
1459
size_t namelen= plugin_name.size(), optnamelen;
1865
const char *plugin_name= tmp->getManifest().name;
1866
uint32_t namelen= strlen(plugin_name), optnamelen;
1867
uint32_t buffer_length= namelen * 4 + (can_disable ? 75 : 10);
1868
char *name= (char*) alloc_root(mem_root, buffer_length) + 1;
1461
1869
char *optname, *p;
1462
1870
int index= 0, offset= 0;
1463
drizzle_sys_var *opt, **plugin_option;
1871
st_mysql_sys_var *opt, **plugin_option;
1464
1872
st_bookmark *v;
1466
string name(plugin_name);
1467
transform(name.begin(), name.end(), name.begin(), ::tolower);
1469
for (string::iterator iter= name.begin(); iter != name.end(); ++iter)
1874
/* support --skip-plugin-foo syntax */
1875
memcpy(name, plugin_name, namelen + 1);
1876
my_casedn_str(&my_charset_utf8_general_ci, name);
1877
sprintf(name+namelen+1, "plugin-%s", name);
1878
/* Now we have namelen + 1 + 7 + namelen + 1 == namelen * 2 + 9. */
1880
for (p= name + namelen*2 + 8; p > name; p--)
1886
sprintf(name+namelen*2+10,
1887
"Enable %s plugin. Disable with --skip-%s (will save memory).",
1890
Now we have namelen * 2 + 10 (one char unused) + 7 + namelen + 9 +
1891
20 + namelen + 20 + 1 == namelen * 4 + 67.
1894
options[0].comment= name + namelen*2 + 10;
1897
options[1].name= (options[0].name= name) + namelen + 1;
1898
options[0].id= options[1].id= 256; /* must be >255. dup id ok */
1899
options[0].var_type= options[1].var_type= GET_BOOL;
1900
options[0].arg_type= options[1].arg_type= NO_ARG;
1901
options[0].def_value= options[1].def_value= true;
1902
options[0].value= options[0].u_max_value=
1903
options[1].value= options[1].u_max_value= (char**) (name - 1);
1476
1907
Two passes as the 2nd pass will take pointer addresses for use
1477
1908
by my_getopt and register_var() in the first pass uses realloc