12
12
You should have received a copy of the GNU General Public License
13
13
along with this program; if not, write to the Free Software
14
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
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/configvar.h>
24
#include <drizzled/qcache.h>
25
#include <drizzled/parser.h>
26
#include <drizzled/sql_parse.h>
27
#include <drizzled/scheduling.h>
28
#include <drizzled/replicator.h>
29
#include <drizzled/show.h>
30
#include <drizzled/handler.h>
31
#include <drizzled/set_var.h>
32
#include <drizzled/session.h>
33
#include <drizzled/item/null.h>
27
#include <boost/program_options.hpp>
29
#include "drizzled/option.h"
30
#include "drizzled/internal/m_string.h"
32
#include "drizzled/plugin.h"
33
#include "drizzled/module/load_list.h"
34
#include "drizzled/module/library.h"
35
#include "drizzled/module/registry.h"
36
#include "drizzled/module/option_context.h"
37
#include "drizzled/sql_parse.h"
38
#include "drizzled/show.h"
39
#include "drizzled/cursor.h"
40
#include "drizzled/set_var.h"
41
#include "drizzled/session.h"
42
#include "drizzled/item/null.h"
43
#include "drizzled/error.h"
44
#include "drizzled/gettext.h"
45
#include "drizzled/errmsg_print.h"
46
#include "drizzled/strfunc.h"
47
#include "drizzled/pthread_globals.h"
48
#include "drizzled/util/tokenize.h"
50
#include <boost/foreach.hpp>
52
/* FreeBSD 2.2.2 does not define RTLD_NOW) */
57
namespace po=boost::program_options;
38
#include <drizzled/error.h>
39
#include <drizzled/gettext.h>
41
#define REPORT_TO_LOG 1
42
#define REPORT_TO_USER 2
44
#define plugin_ref_to_int(A) (A ? A[0] : NULL)
45
#define plugin_int_to_ref(A) &(A)
59
47
using namespace std;
61
/** These exist just to prevent symbols from being optimized out */
62
typedef drizzled::module::Manifest drizzled_builtin_list[];
63
extern drizzled_builtin_list PANDORA_BUILTIN_SYMBOLS_LIST;
64
extern drizzled_builtin_list PANDORA_BUILTIN_LOAD_SYMBOLS_LIST;
65
drizzled::module::Manifest *drizzled_builtins[]=
67
PANDORA_BUILTIN_SYMBOLS_LIST, NULL
69
drizzled::module::Manifest *drizzled_load_builtins[]=
71
PANDORA_BUILTIN_LOAD_SYMBOLS_LIST, NULL
78
typedef vector<string> PluginOptions;
79
static PluginOptions opt_plugin_load;
80
static PluginOptions opt_plugin_add;
81
static PluginOptions opt_plugin_remove;
82
const char *builtin_plugins= PANDORA_BUILTIN_LIST;
83
const char *builtin_load_plugins= PANDORA_BUILTIN_LOAD_LIST;
49
extern struct st_mysql_plugin *mysqld_builtins[];
51
char *opt_plugin_load= NULL;
52
char *opt_plugin_dir_ptr;
53
char opt_plugin_dir[FN_REFLEN];
55
When you ad a new plugin type, add both a string and make sure that the
56
init and deinit array are correctly updated.
58
const LEX_STRING plugin_type_names[DRIZZLE_MAX_PLUGIN_TYPE_NUM]=
60
{ C_STRING_WITH_LEN("DAEMON") },
61
{ C_STRING_WITH_LEN("STORAGE ENGINE") },
62
{ C_STRING_WITH_LEN("INFORMATION SCHEMA") },
63
{ C_STRING_WITH_LEN("UDF") },
64
{ C_STRING_WITH_LEN("UDA") },
65
{ C_STRING_WITH_LEN("AUDIT") },
66
{ C_STRING_WITH_LEN("LOGGER") },
67
{ C_STRING_WITH_LEN("ERRMSG") },
68
{ C_STRING_WITH_LEN("AUTH") },
69
{ C_STRING_WITH_LEN("CONFIGVAR") },
70
{ C_STRING_WITH_LEN("QCACHE") },
71
{ C_STRING_WITH_LEN("PARSER") },
72
{ C_STRING_WITH_LEN("SCHEDULING") },
73
{ C_STRING_WITH_LEN("REPLICATOR") }
76
extern int initialize_schema_table(st_plugin_int *plugin);
77
extern int finalize_schema_table(st_plugin_int *plugin);
79
extern int initialize_udf(st_plugin_int *plugin);
80
extern int finalize_udf(st_plugin_int *plugin);
83
The number of elements in both plugin_type_initialize and
84
plugin_type_deinitialize should equal to the number of plugins
87
plugin_type_init plugin_type_initialize[DRIZZLE_MAX_PLUGIN_TYPE_NUM]=
90
ha_initialize_handlerton, /* Storage Engine */
91
initialize_schema_table, /* Information Schema */
92
initialize_udf, /* UDF */
95
logging_initializer, /* Logger */
96
errmsg_initializer, /* Error Messages */
97
authentication_initializer, /* Auth */
98
configvar_initializer,
101
scheduling_initializer,
102
replicator_initializer
105
plugin_type_init plugin_type_deinitialize[DRIZZLE_MAX_PLUGIN_TYPE_NUM]=
108
ha_finalize_handlerton, /* Storage Engine */
109
finalize_schema_table, /* Information Schema */
110
finalize_udf, /* UDF */
113
logging_finalizer, /* Logger */
114
errmsg_finalizer, /* Logger */
115
authentication_finalizer, /* Auth */
119
scheduling_finalizer,
123
static const char *plugin_declarations_sym= "_mysql_plugin_declarations_";
85
125
/* Note that 'int version' must be the first field of every plugin
86
126
sub-structure (plugin->info).
89
static bool initialized= false;
129
static bool initialized= 0;
131
static DYNAMIC_ARRAY plugin_dl_array;
132
static DYNAMIC_ARRAY plugin_array;
133
static HASH plugin_hash[DRIZZLE_MAX_PLUGIN_TYPE_NUM];
92
134
static bool reap_needed= false;
135
static int plugin_array_version=0;
95
138
write-lock on LOCK_system_variables_hash is required before modifying
96
139
the following variables/structures
98
static memory::Root plugin_mem_root(4096);
141
static MEM_ROOT plugin_mem_root;
99
142
static uint32_t global_variables_dynamic_size= 0;
143
static HASH bookmark_hash;
103
147
hidden part of opaque value passed to variable check functions.
104
148
Used to provide a object-like structure to non C++ consumers.
106
struct st_item_value_holder : public drizzle_value
150
struct st_item_value_holder : public st_mysql_value
157
stored in bookmark_hash, this structure is never removed from the
158
hash and is used to mark a single offset for a session local variable
159
even if plugins have been uninstalled and reinstalled, repeatedly.
160
This structure is allocated from plugin_mem_root.
162
The key format is as follows:
163
1 byte - variable type code
164
name_len bytes - variable name
122
171
uint32_t version;
126
typedef boost::unordered_map<string, Bookmark> bookmark_unordered_map;
127
static bookmark_unordered_map bookmark_hash;
177
skeleton of a plugin variable - portion of structure common to all.
179
struct st_mysql_sys_var
181
DRIZZLE_PLUGIN_VAR_HEADER;
186
sys_var class for access to all plugin variables visible to the user
188
class sys_var_pluginvar: public sys_var
191
struct st_plugin_int *plugin;
192
struct st_mysql_sys_var *plugin_var;
194
static void *operator new(size_t size, MEM_ROOT *mem_root)
195
{ return (void*) alloc_root(mem_root, (uint) size); }
196
static void operator delete(void *, size_t)
197
{ TRASH(ptr_arg, size); }
199
sys_var_pluginvar(const char *name_arg,
200
struct st_mysql_sys_var *plugin_var_arg)
201
:sys_var(name_arg), plugin_var(plugin_var_arg) {}
202
sys_var_pluginvar *cast_pluginvar() { return this; }
203
bool is_readonly() const { return plugin_var->flags & PLUGIN_VAR_READONLY; }
204
bool check_type(enum_var_type type)
205
{ return !(plugin_var->flags & PLUGIN_VAR_SessionLOCAL) && type != OPT_GLOBAL; }
206
bool check_update_type(Item_result type);
207
SHOW_TYPE show_type();
208
unsigned char* real_value_ptr(Session *session, enum_var_type type);
209
TYPELIB* plugin_var_typelib(void);
210
unsigned char* value_ptr(Session *session, enum_var_type type, LEX_STRING *base);
211
bool check(Session *session, set_var *var);
212
bool check_default(enum_var_type)
213
{ return is_readonly(); }
214
void set_default(Session *session, enum_var_type);
215
bool update(Session *session, set_var *var);
132
static void plugin_prune_list(vector<string> &plugin_list,
133
const vector<string> &plugins_to_remove);
134
static bool plugin_load_list(module::Registry ®istry,
135
memory::Root *tmp_root,
136
const set<string> &plugin_list,
137
po::options_description &long_options,
138
bool builtin= false);
139
static int test_plugin_options(memory::Root *, module::Module *,
140
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);
220
static bool plugin_load_list(MEM_ROOT *tmp_root, int *argc, char **argv,
222
static int test_plugin_options(MEM_ROOT *, struct st_plugin_int *,
224
static bool register_builtin(struct st_mysql_plugin *, struct st_plugin_int *,
225
struct st_plugin_int **);
226
static void unlock_variables(Session *session, struct system_variables *vars);
227
static void cleanup_variables(Session *session, struct system_variables *vars);
228
static void plugin_vars_free_values(sys_var *vars);
229
static void plugin_opt_set_limits(struct my_option *options,
230
const struct st_mysql_sys_var *opt);
231
#define my_intern_plugin_lock(A,B) intern_plugin_lock(A,B)
232
#define my_intern_plugin_lock_ci(A,B) intern_plugin_lock(A,B)
233
static plugin_ref intern_plugin_lock(LEX *lex, plugin_ref plugin);
234
static void intern_plugin_unlock(LEX *lex, plugin_ref plugin);
235
static void reap_plugins(void);
238
/* declared in set_var.cc */
239
extern sys_var *intern_find_sys_var(const char *str, uint32_t length, bool no_error);
240
extern bool throw_bounds_warning(Session *session, bool fixed, bool unsignd,
241
const char *name, int64_t val);
243
/****************************************************************************
244
Value type thunks, allows the C world to play in the C++ world
245
****************************************************************************/
247
static int item_value_type(struct st_mysql_value *value)
249
switch (((st_item_value_holder*)value)->item->result_type()) {
251
return DRIZZLE_VALUE_TYPE_INT;
253
return DRIZZLE_VALUE_TYPE_REAL;
255
return DRIZZLE_VALUE_TYPE_STRING;
259
static const char *item_val_str(struct st_mysql_value *value,
260
char *buffer, int *length)
262
String str(buffer, *length, system_charset_info), *res;
263
if (!(res= ((st_item_value_holder*)value)->item->val_str(&str)))
265
*length= res->length();
266
if (res->c_ptr_quick() == buffer)
270
Lets be nice and create a temporary string since the
273
return current_session->strmake(res->c_ptr_quick(), res->length());
277
static int item_val_int(struct st_mysql_value *value, int64_t *buf)
279
Item *item= ((st_item_value_holder*)value)->item;
280
*buf= item->val_int();
287
static int item_val_real(struct st_mysql_value *value, double *buf)
289
Item *item= ((st_item_value_holder*)value)->item;
290
*buf= item->val_real();
145
297
/****************************************************************************
146
298
Plugin support code
147
299
****************************************************************************/
301
static struct st_plugin_dl *plugin_dl_find(const LEX_STRING *dl)
304
struct st_plugin_dl *tmp;
306
for (i= 0; i < plugin_dl_array.elements; i++)
308
tmp= *dynamic_element(&plugin_dl_array, i, struct st_plugin_dl **);
309
if (tmp->ref_count &&
310
! my_strnncoll(files_charset_info,
311
(const unsigned char *)dl->str, dl->length,
312
(const unsigned char *)tmp->dl.str, tmp->dl.length))
318
static st_plugin_dl *plugin_dl_insert_or_reuse(struct st_plugin_dl *plugin_dl)
321
struct st_plugin_dl *tmp;
323
for (i= 0; i < plugin_dl_array.elements; i++)
325
tmp= *dynamic_element(&plugin_dl_array, i, struct st_plugin_dl **);
326
if (! tmp->ref_count)
328
memcpy(tmp, plugin_dl, sizeof(struct st_plugin_dl));
332
if (insert_dynamic(&plugin_dl_array, (unsigned char*)&plugin_dl))
334
tmp= *dynamic_element(&plugin_dl_array, plugin_dl_array.elements - 1,
335
struct st_plugin_dl **)=
336
(struct st_plugin_dl *) memdup_root(&plugin_mem_root, (unsigned char*)plugin_dl,
337
sizeof(struct st_plugin_dl));
341
static inline void free_plugin_mem(struct st_plugin_dl *p)
349
static st_plugin_dl *plugin_dl_add(const LEX_STRING *dl, int report)
352
uint32_t plugin_dir_len, dummy_errors;
353
struct st_plugin_dl *tmp, plugin_dl;
355
plugin_dir_len= strlen(opt_plugin_dir);
356
dlpath.reserve(FN_REFLEN);
358
Ensure that the dll doesn't have a path.
359
This is done to ensure that only approved libraries from the
360
plugin directory are used (to make this even remotely secure).
362
if (strchr(dl->str, FN_LIBCHAR) ||
363
check_string_char_length((LEX_STRING *) dl, "", NAME_CHAR_LEN,
364
system_charset_info, 1) ||
365
plugin_dir_len + dl->length + 1 >= FN_REFLEN)
367
if (report & REPORT_TO_USER)
368
my_error(ER_UDF_NO_PATHS, MYF(0));
369
if (report & REPORT_TO_LOG)
370
errmsg_printf(ERRMSG_LVL_ERROR, "%s",ER(ER_UDF_NO_PATHS));
373
/* If this dll is already loaded just increase ref_count. */
374
if ((tmp= plugin_dl_find(dl)))
379
memset(&plugin_dl, 0, sizeof(plugin_dl));
380
/* Compile dll path */
381
dlpath.append(opt_plugin_dir);
383
dlpath.append(dl->str);
384
plugin_dl.ref_count= 1;
385
/* Open new dll handle */
386
if (!(plugin_dl.handle= dlopen(dlpath.c_str(), RTLD_LAZY|RTLD_GLOBAL)))
388
const char *errmsg=dlerror();
389
uint32_t dlpathlen= dlpath.length();
390
if (!dlpath.compare(0, dlpathlen, errmsg))
391
{ // if errmsg starts from dlpath, trim this prefix.
393
if (*errmsg == ':') errmsg++;
394
if (*errmsg == ' ') errmsg++;
396
if (report & REPORT_TO_USER)
397
my_error(ER_CANT_OPEN_LIBRARY, MYF(0), dlpath.c_str(), errno, errmsg);
398
if (report & REPORT_TO_LOG)
399
errmsg_printf(ERRMSG_LVL_ERROR, ER(ER_CANT_OPEN_LIBRARY), dlpath.c_str(), errno, errmsg);
403
/* Find plugin declarations */
404
if (!(sym= dlsym(plugin_dl.handle, plugin_declarations_sym)))
406
free_plugin_mem(&plugin_dl);
407
if (report & REPORT_TO_USER)
408
my_error(ER_CANT_FIND_DL_ENTRY, MYF(0), plugin_declarations_sym);
409
if (report & REPORT_TO_LOG)
410
errmsg_printf(ERRMSG_LVL_ERROR, ER(ER_CANT_FIND_DL_ENTRY), plugin_declarations_sym);
414
plugin_dl.plugins= (struct st_mysql_plugin *)sym;
416
/* Duplicate and convert dll name */
417
plugin_dl.dl.length= dl->length * files_charset_info->mbmaxlen + 1;
418
if (! (plugin_dl.dl.str= (char*) malloc(plugin_dl.dl.length)))
420
free_plugin_mem(&plugin_dl);
421
if (report & REPORT_TO_USER)
422
my_error(ER_OUTOFMEMORY, MYF(0), plugin_dl.dl.length);
423
if (report & REPORT_TO_LOG)
424
errmsg_printf(ERRMSG_LVL_ERROR, ER(ER_OUTOFMEMORY), plugin_dl.dl.length);
427
plugin_dl.dl.length= copy_and_convert(plugin_dl.dl.str, plugin_dl.dl.length,
428
files_charset_info, dl->str, dl->length, system_charset_info,
430
plugin_dl.dl.str[plugin_dl.dl.length]= 0;
431
/* Add this dll to array */
432
if (! (tmp= plugin_dl_insert_or_reuse(&plugin_dl)))
434
free_plugin_mem(&plugin_dl);
435
if (report & REPORT_TO_USER)
436
my_error(ER_OUTOFMEMORY, MYF(0), sizeof(struct st_plugin_dl));
437
if (report & REPORT_TO_LOG)
438
errmsg_printf(ERRMSG_LVL_ERROR, ER(ER_OUTOFMEMORY), sizeof(struct st_plugin_dl));
445
static void plugin_dl_del(const LEX_STRING *dl)
449
for (i= 0; i < plugin_dl_array.elements; i++)
451
struct st_plugin_dl *tmp= *dynamic_element(&plugin_dl_array, i,
452
struct st_plugin_dl **);
453
if (tmp->ref_count &&
454
! my_strnncoll(files_charset_info,
455
(const unsigned char *)dl->str, dl->length,
456
(const unsigned char *)tmp->dl.str, tmp->dl.length))
458
/* Do not remove this element, unless no other plugin uses this dll. */
459
if (! --tmp->ref_count)
461
free_plugin_mem(tmp);
462
memset(tmp, 0, sizeof(struct st_plugin_dl));
471
static struct st_plugin_int *plugin_find_internal(const LEX_STRING *name, int type)
477
if (type == DRIZZLE_ANY_PLUGIN)
479
for (i= 0; i < DRIZZLE_MAX_PLUGIN_TYPE_NUM; i++)
481
struct st_plugin_int *plugin= (st_plugin_int *)
482
hash_search(&plugin_hash[i], (const unsigned char *)name->str, name->length);
488
return((st_plugin_int *)
489
hash_search(&plugin_hash[type], (const unsigned char *)name->str, name->length));
494
static SHOW_COMP_OPTION plugin_status(const LEX_STRING *name, int type)
496
SHOW_COMP_OPTION rc= SHOW_OPTION_NO;
497
struct st_plugin_int *plugin;
498
if ((plugin= plugin_find_internal(name, type)))
500
rc= SHOW_OPTION_DISABLED;
501
if (plugin->state == PLUGIN_IS_READY)
508
bool plugin_is_ready(const LEX_STRING *name, int type)
511
if (plugin_status(name, type) == SHOW_OPTION_YES)
517
SHOW_COMP_OPTION sys_var_have_plugin::get_option()
519
LEX_STRING plugin_name= { (char *) plugin_name_str, plugin_name_len };
520
return plugin_status(&plugin_name, plugin_type);
524
static plugin_ref intern_plugin_lock(LEX *, plugin_ref rc)
526
st_plugin_int *pi= plugin_ref_to_int(rc);
528
if (pi->state & (PLUGIN_IS_READY | PLUGIN_IS_UNINITIALIZED))
532
For debugging, we do an additional malloc which allows the
533
memory manager and/or valgrind to track locked references and
534
double unlocks to aid resolving reference counting.problems.
536
if (!(plugin= (plugin_ref) malloc(sizeof(pi))))
548
plugin_ref plugin_lock(Session *session, plugin_ref *ptr)
550
LEX *lex= session ? session->lex : 0;
552
rc= my_intern_plugin_lock_ci(lex, *ptr);
557
plugin_ref plugin_lock_by_name(Session *session, const LEX_STRING *name, int type)
559
LEX *lex= session ? session->lex : 0;
561
st_plugin_int *plugin;
562
if ((plugin= plugin_find_internal(name, type)))
563
rc= my_intern_plugin_lock_ci(lex, plugin_int_to_ref(plugin));
568
static st_plugin_int *plugin_insert_or_reuse(struct st_plugin_int *plugin)
571
struct st_plugin_int *tmp;
572
for (i= 0; i < plugin_array.elements; i++)
574
tmp= *dynamic_element(&plugin_array, i, struct st_plugin_int **);
575
if (tmp->state == PLUGIN_IS_FREED)
577
memcpy(tmp, plugin, sizeof(struct st_plugin_int));
581
if (insert_dynamic(&plugin_array, (unsigned char*)&plugin))
583
tmp= *dynamic_element(&plugin_array, plugin_array.elements - 1,
584
struct st_plugin_int **)=
585
(struct st_plugin_int *) memdup_root(&plugin_mem_root, (unsigned char*)plugin,
586
sizeof(struct st_plugin_int));
154
593
Requires that a write-lock is held on LOCK_system_variables_hash
156
static bool plugin_add(module::Registry ®istry, memory::Root *tmp_root,
157
module::Library *library,
158
po::options_description &long_options)
595
static bool plugin_add(MEM_ROOT *tmp_root,
596
const LEX_STRING *name, const LEX_STRING *dl,
597
int *argc, char **argv, int report)
163
if (registry.find(library->getName()))
599
struct st_plugin_int tmp;
600
struct st_mysql_plugin *plugin;
601
if (plugin_find_internal(name, DRIZZLE_ANY_PLUGIN))
165
errmsg_printf(error::WARN, ER(ER_PLUGIN_EXISTS),
166
library->getName().c_str());
603
if (report & REPORT_TO_USER)
604
my_error(ER_UDF_EXISTS, MYF(0), name->str);
605
if (report & REPORT_TO_LOG)
606
errmsg_printf(ERRMSG_LVL_ERROR, ER(ER_UDF_EXISTS), name->str);
170
module::Module *tmp= NULL;
609
/* Clear the whole struct to catch future extensions. */
610
memset(&tmp, 0, sizeof(tmp));
611
if (! (tmp.plugin_dl= plugin_dl_add(dl, report)))
171
613
/* Find plugin by name */
172
const module::Manifest *manifest= library->getManifest();
174
if (registry.find(manifest->name))
176
errmsg_printf(error::ERROR,
177
_("Plugin '%s' contains the name '%s' in its manifest, which "
178
"has already been registered.\n"),
179
library->getName().c_str(),
184
tmp= new (std::nothrow) module::Module(manifest, library);
188
if (!test_plugin_options(tmp_root, tmp, long_options))
193
errmsg_printf(error::ERROR, ER(ER_CANT_FIND_DL_ENTRY),
194
library->getName().c_str());
199
static void reap_plugins(module::Registry ®istry)
201
std::map<std::string, module::Module *>::const_iterator modules=
202
registry.getModulesMap().begin();
204
while (modules != registry.getModulesMap().end())
206
module::Module *module= (*modules).second;
213
static bool plugin_initialize(module::Registry ®istry,
214
module::Module *module)
216
assert(module->isInited == false);
218
module::Context loading_context(registry, module);
219
if (module->getManifest().init)
221
if (module->getManifest().init(loading_context))
223
errmsg_printf(error::ERROR,
224
_("Plugin '%s' init function returned error.\n"),
225
module->getName().c_str());
229
module->isInited= true;
236
inline static void dashes_to_underscores(std::string &name_in,
237
char from= '-', char to= '_')
239
for (string::iterator p= name_in.begin();
250
inline static void underscores_to_dashes(std::string &name_in)
252
return dashes_to_underscores(name_in, '_', '-');
255
static void compose_plugin_options(vector<string> &target,
256
vector<string> options)
258
for (vector<string>::iterator it= options.begin();
262
tokenize(*it, target, ",", true);
264
for (vector<string>::iterator it= target.begin();
268
dashes_to_underscores(*it);
272
void compose_plugin_add(vector<string> options)
274
compose_plugin_options(opt_plugin_add, options);
277
void compose_plugin_remove(vector<string> options)
279
compose_plugin_options(opt_plugin_remove, options);
282
void notify_plugin_load(string in_plugin_load)
284
tokenize(in_plugin_load, opt_plugin_load, ",", true);
614
for (plugin= tmp.plugin_dl->plugins; plugin->name; plugin++)
616
uint32_t name_len= strlen(plugin->name);
617
if (plugin->type < DRIZZLE_MAX_PLUGIN_TYPE_NUM &&
618
! my_strnncoll(system_charset_info,
619
(const unsigned char *)name->str, name->length,
620
(const unsigned char *)plugin->name,
623
struct st_plugin_int *tmp_plugin_ptr;
626
tmp.name.str= (char *)plugin->name;
627
tmp.name.length= name_len;
629
tmp.state= PLUGIN_IS_UNINITIALIZED;
630
if (!test_plugin_options(tmp_root, &tmp, argc, argv))
632
if ((tmp_plugin_ptr= plugin_insert_or_reuse(&tmp)))
634
plugin_array_version++;
635
if (!my_hash_insert(&plugin_hash[plugin->type], (unsigned char*)tmp_plugin_ptr))
637
init_alloc_root(&tmp_plugin_ptr->mem_root, 4096, 4096);
640
tmp_plugin_ptr->state= PLUGIN_IS_FREED;
642
mysql_del_sys_var_chain(tmp.system_vars);
645
/* plugin was disabled */
650
if (report & REPORT_TO_USER)
651
my_error(ER_CANT_FIND_DL_ENTRY, MYF(0), name->str);
652
if (report & REPORT_TO_LOG)
653
errmsg_printf(ERRMSG_LVL_ERROR, ER(ER_CANT_FIND_DL_ENTRY), name->str);
660
static void plugin_deinitialize(struct st_plugin_int *plugin, bool ref_check)
662
if (plugin->plugin->status_vars)
666
We have a problem right now where we can not prepend without
667
breaking backwards compatibility. We will fix this shortly so
668
that engines have "use names" and we wil use those for
669
CREATE TABLE, and use the plugin name then for adding automatic
673
{plugin->plugin->name, (char*)plugin->plugin->status_vars, SHOW_ARRAY},
676
remove_status_vars(array);
678
remove_status_vars(plugin->plugin->status_vars);
679
#endif /* FIX_LATER */
682
if (plugin_type_deinitialize[plugin->plugin->type])
684
if ((*plugin_type_deinitialize[plugin->plugin->type])(plugin))
686
errmsg_printf(ERRMSG_LVL_ERROR, _("Plugin '%s' of type %s failed deinitialization"),
687
plugin->name.str, plugin_type_names[plugin->plugin->type].str);
690
else if (plugin->plugin->deinit)
691
plugin->plugin->deinit(plugin);
693
plugin->state= PLUGIN_IS_UNINITIALIZED;
696
We do the check here because NDB has a worker Session which doesn't
697
exit until NDB is shut down.
699
if (ref_check && plugin->ref_count)
700
errmsg_printf(ERRMSG_LVL_ERROR, _("Plugin '%s' has ref_count=%d after deinitialization."),
701
plugin->name.str, plugin->ref_count);
705
static void plugin_del(struct st_plugin_int *plugin)
707
/* Free allocated strings before deleting the plugin. */
708
plugin_vars_free_values(plugin->system_vars);
709
hash_delete(&plugin_hash[plugin->plugin->type], (unsigned char*)plugin);
710
if (plugin->plugin_dl)
711
plugin_dl_del(&plugin->plugin_dl->dl);
712
plugin->state= PLUGIN_IS_FREED;
713
plugin_array_version++;
714
pthread_rwlock_wrlock(&LOCK_system_variables_hash);
715
mysql_del_sys_var_chain(plugin->system_vars);
716
pthread_rwlock_unlock(&LOCK_system_variables_hash);
717
free_root(&plugin->mem_root, MYF(0));
721
static void reap_plugins(void)
725
struct st_plugin_int *plugin;
728
count= plugin_array.elements;
730
for (idx= 0; idx < count; idx++)
732
plugin= *dynamic_element(&plugin_array, idx, struct st_plugin_int **);
733
plugin->state= PLUGIN_IS_DYING;
734
plugin_deinitialize(plugin, true);
739
static void intern_plugin_unlock(LEX *, plugin_ref plugin)
746
pi= plugin_ref_to_int(plugin);
748
free((void *) plugin);
750
assert(pi->ref_count);
753
if (pi->state == PLUGIN_IS_DELETED && !pi->ref_count)
760
void plugin_unlock(Session *session, plugin_ref plugin)
762
LEX *lex= session ? session->lex : 0;
765
intern_plugin_unlock(lex, plugin);
770
void plugin_unlock_list(Session *session, plugin_ref *list, uint32_t count)
772
LEX *lex= session ? session->lex : 0;
775
intern_plugin_unlock(lex, *list++);
780
static int plugin_initialize(struct st_plugin_int *plugin)
783
if (plugin_type_initialize[plugin->plugin->type])
785
if ((*plugin_type_initialize[plugin->plugin->type])(plugin))
787
errmsg_printf(ERRMSG_LVL_ERROR, _("Plugin '%s' registration as a %s failed."),
788
plugin->name.str, plugin_type_names[plugin->plugin->type].str);
792
else if (plugin->plugin->init)
794
if (plugin->plugin->init(plugin))
796
errmsg_printf(ERRMSG_LVL_ERROR, _("Plugin '%s' init function returned error."),
802
plugin->state= PLUGIN_IS_READY;
804
if (plugin->plugin->status_vars)
808
We have a problem right now where we can not prepend without
809
breaking backwards compatibility. We will fix this shortly so
810
that engines have "use names" and we wil use those for
811
CREATE TABLE, and use the plugin name then for adding automatic
815
{plugin->plugin->name, (char*)plugin->plugin->status_vars, SHOW_ARRAY},
818
if (add_status_vars(array)) // add_status_vars makes a copy
821
add_status_vars(plugin->plugin->status_vars); // add_status_vars makes a copy
822
#endif /* FIX_LATER */
826
set the plugin attribute of plugin's sys vars so they are pointing
829
if (plugin->system_vars)
831
sys_var_pluginvar *var= plugin->system_vars->cast_pluginvar();
837
var= var->next->cast_pluginvar();
847
extern "C" unsigned char *get_plugin_hash_key(const unsigned char *, size_t *, bool);
848
extern "C" unsigned char *get_bookmark_hash_key(const unsigned char *, size_t *, bool);
851
unsigned char *get_plugin_hash_key(const unsigned char *buff, size_t *length, bool)
853
struct st_plugin_int *plugin= (st_plugin_int *)buff;
854
*length= (uint)plugin->name.length;
855
return((unsigned char *)plugin->name.str);
859
unsigned char *get_bookmark_hash_key(const unsigned char *buff, size_t *length, bool)
861
struct st_bookmark *var= (st_bookmark *)buff;
862
*length= var->name_len + 1;
863
return (unsigned char*) var->key;
288
868
The logic is that we first load and initialize all compiled in plugins.
292
872
Finally we initialize everything, aka the dynamic that have yet to initialize.
294
bool plugin_init(module::Registry ®istry,
295
po::options_description &long_options)
874
int plugin_init(int *argc, char **argv, int flags)
297
memory::Root tmp_root(4096);
877
struct st_mysql_plugin **builtins;
878
struct st_mysql_plugin *plugin;
879
struct st_plugin_int tmp, *plugin_ptr;
885
init_alloc_root(&plugin_mem_root, 4096, 4096);
886
init_alloc_root(&tmp_root, 4096, 4096);
888
if (hash_init(&bookmark_hash, &my_charset_bin, 16, 0, 0,
889
get_bookmark_hash_key, NULL, HASH_UNIQUE))
893
if (my_init_dynamic_array(&plugin_dl_array,
894
sizeof(struct st_plugin_dl *),16,16) ||
895
my_init_dynamic_array(&plugin_array,
896
sizeof(struct st_plugin_int *),16,16))
899
for (idx= 0; idx < DRIZZLE_MAX_PLUGIN_TYPE_NUM; idx++)
901
if (hash_init(&plugin_hash[idx], system_charset_info, 16, 0, 0,
902
get_plugin_hash_key, NULL, HASH_UNIQUE))
304
PluginOptions builtin_load_list;
305
tokenize(builtin_load_plugins, builtin_load_list, ",", true);
307
PluginOptions builtin_list;
308
tokenize(builtin_plugins, builtin_list, ",", true);
310
bool load_failed= false;
312
if (opt_plugin_add.size() > 0)
314
for (PluginOptions::iterator iter= opt_plugin_add.begin();
315
iter != opt_plugin_add.end();
318
if (find(builtin_list.begin(),
319
builtin_list.end(), *iter) != builtin_list.end())
321
builtin_load_list.push_back(*iter);
325
opt_plugin_load.push_back(*iter);
330
if (opt_plugin_remove.size() > 0)
332
plugin_prune_list(opt_plugin_load, opt_plugin_remove);
333
plugin_prune_list(builtin_load_list, opt_plugin_remove);
338
909
First we register builtin plugins
340
const set<string> builtin_list_set(builtin_load_list.begin(),
341
builtin_load_list.end());
342
load_failed= plugin_load_list(registry, &tmp_root,
343
builtin_list_set, long_options, true);
911
for (builtins= mysqld_builtins; *builtins; builtins++)
346
tmp_root.free_root(MYF(0));
913
for (plugin= *builtins; plugin->name; plugin++)
915
memset(&tmp, 0, sizeof(tmp));
917
tmp.name.str= (char *)plugin->name;
918
tmp.name.length= strlen(plugin->name);
920
free_root(&tmp_root, MYF(MY_MARK_BLOCKS_FREE));
921
if (test_plugin_options(&tmp_root, &tmp, argc, argv))
924
if (register_builtin(plugin, &tmp, &plugin_ptr))
927
if (plugin_initialize(plugin_ptr))
931
initialize the global default storage engine so that it may
932
not be null in any child thread.
934
if (my_strcasecmp(&my_charset_utf8_general_ci, plugin->name, "MyISAM") == 0)
936
assert(!global_system_variables.table_plugin);
937
global_system_variables.table_plugin=
938
my_intern_plugin_lock(NULL, plugin_int_to_ref(plugin_ptr));
939
assert(plugin_ptr->ref_count == 1);
350
/* Uniquify the list */
351
const set<string> plugin_list_set(opt_plugin_load.begin(),
352
opt_plugin_load.end());
944
/* should now be set to MyISAM storage engine */
945
assert(global_system_variables.table_plugin);
354
947
/* Register all dynamic plugins */
355
load_failed= plugin_load_list(registry, &tmp_root,
356
plugin_list_set, long_options);
948
if (!(flags & PLUGIN_INIT_SKIP_DYNAMIC_LOADING))
359
tmp_root.free_root(MYF(0));
951
plugin_load_list(&tmp_root, argc, argv, opt_plugin_load);
363
tmp_root.free_root(MYF(0));
368
bool plugin_finalize(module::Registry ®istry)
954
if (flags & PLUGIN_INIT_SKIP_INITIALIZATION)
371
958
Now we initialize all remaining plugins
373
module::Registry::ModuleList module_list= registry.getList();
374
module::Registry::ModuleList::iterator modules= module_list.begin();
376
while (modules != module_list.end())
960
for (idx= 0; idx < plugin_array.elements; idx++)
378
module::Module *module= *modules;
380
if (module->isInited == false)
962
plugin_ptr= *dynamic_element(&plugin_array, idx, struct st_plugin_int **);
963
if (plugin_ptr->state == PLUGIN_IS_UNINITIALIZED)
382
if (plugin_initialize(registry, module))
965
if (plugin_initialize(plugin_ptr))
384
registry.remove(module);
967
plugin_ptr->state= PLUGIN_IS_DYING;
968
plugin_deinitialize(plugin_ptr, true);
969
plugin_del(plugin_ptr);
392
BOOST_FOREACH(plugin::Plugin::map::value_type value, registry.getPluginsMap())
394
value.second->prime();
401
Window of opportunity for plugins to issue any queries with the database up and running but with no user's connected.
403
void plugin_startup_window(module::Registry ®istry, drizzled::Session &session)
405
BOOST_FOREACH(plugin::Plugin::map::value_type value, registry.getPluginsMap())
407
value.second->startup(session);
412
public unary_function<string, bool>
414
const string to_match;
416
PrunePlugin& operator=(const PrunePlugin&);
418
explicit PrunePlugin(const string &match_in) :
422
result_type operator()(const string &match_against)
424
return match_against == to_match;
428
static void plugin_prune_list(vector<string> &plugin_list,
429
const vector<string> &plugins_to_remove)
431
for (vector<string>::const_iterator iter= plugins_to_remove.begin();
432
iter != plugins_to_remove.end();
435
plugin_list.erase(remove_if(plugin_list.begin(),
976
free_root(&tmp_root, MYF(0));
982
free_root(&tmp_root, MYF(0));
987
static bool register_builtin(struct st_mysql_plugin *plugin,
988
struct st_plugin_int *tmp,
989
struct st_plugin_int **ptr)
992
tmp->state= PLUGIN_IS_UNINITIALIZED;
996
if (insert_dynamic(&plugin_array, (unsigned char*)&tmp))
999
*ptr= *dynamic_element(&plugin_array, plugin_array.elements - 1,
1000
struct st_plugin_int **)=
1001
(struct st_plugin_int *) memdup_root(&plugin_mem_root, (unsigned char*)tmp,
1002
sizeof(struct st_plugin_int));
1004
if (my_hash_insert(&plugin_hash[plugin->type],(unsigned char*) *ptr))
443
1012
called only by plugin_init()
445
static bool plugin_load_list(module::Registry ®istry,
446
memory::Root *tmp_root,
447
const set<string> &plugin_list,
448
po::options_description &long_options,
1014
static bool plugin_load_list(MEM_ROOT *tmp_root, int *argc, char **argv,
451
module::Library *library= NULL;
453
for (set<string>::const_iterator iter= plugin_list.begin();
454
iter != plugin_list.end();
1017
char buffer[FN_REFLEN];
1018
LEX_STRING name= {buffer, 0}, dl= {NULL, 0}, *str= &name;
1019
struct st_plugin_dl *plugin_dl;
1020
struct st_mysql_plugin *plugin;
457
const string plugin_name(*iter);
459
library= registry.addLibrary(plugin_name, builtin);
1024
if (p == buffer + sizeof(buffer) - 1)
462
errmsg_printf(error::ERROR,
463
_("Couldn't load plugin library named '%s'.\n"),
464
plugin_name.c_str());
1026
errmsg_printf(ERRMSG_LVL_ERROR, _("plugin-load parameter too long"));
468
tmp_root->free_root(MYF(memory::MARK_BLOCKS_FREE));
469
if (plugin_add(registry, tmp_root, library, long_options))
471
registry.removeLibrary(plugin_name);
472
errmsg_printf(error::ERROR,
473
_("Couldn't load plugin named '%s'.\n"),
474
plugin_name.c_str());
1030
switch ((*(p++)= *(list++))) {
1032
list= NULL; /* terminate the loop */
1034
case ':': /* can't use this as delimiter as it may be drive letter */
1036
str->str[str->length]= '\0';
1037
if (str == &name) // load all plugins in named module
1041
p--; /* reset pointer */
1046
if ((plugin_dl= plugin_dl_add(&dl, REPORT_TO_LOG)))
1048
for (plugin= plugin_dl->plugins; plugin->name; plugin++)
1050
name.str= (char *) plugin->name;
1051
name.length= strlen(name.str);
1053
free_root(tmp_root, MYF(MY_MARK_BLOCKS_FREE));
1054
if (plugin_add(tmp_root, &name, &dl, argc, argv, REPORT_TO_LOG))
1057
plugin_dl_del(&dl); // reduce ref count
1062
free_root(tmp_root, MYF(MY_MARK_BLOCKS_FREE));
1063
if (plugin_add(tmp_root, &name, &dl, argc, argv, REPORT_TO_LOG))
1066
name.length= dl.length= 0;
1067
dl.str= NULL; name.str= p= buffer;
1074
name.str[name.length]= '\0';
1086
errmsg_printf(ERRMSG_LVL_ERROR, _("Couldn't load plugin named '%s' with soname '%s'."),
483
void module_shutdown(module::Registry ®istry)
1092
void plugin_shutdown(void)
1094
uint32_t idx, free_slots= 0;
1095
size_t count= plugin_array.elements;
1096
struct st_plugin_int *plugin;
1097
vector<st_plugin_int *> plugins;
1098
vector<st_plugin_dl *> dl;
486
1100
if (initialized)
488
1102
reap_needed= true;
490
reap_plugins(registry);
491
unlock_variables(NULL, &global_system_variables);
492
unlock_variables(NULL, &max_system_variables);
494
cleanup_variables(&global_system_variables);
495
cleanup_variables(&max_system_variables);
1105
We want to shut down plugins in a reasonable order, this will
1106
become important when we have plugins which depend upon each other.
1107
Circular references cannot be reaped so they are forced afterwards.
1108
TODO: Have an additional step here to notify all active plugins that
1109
shutdown is requested to allow plugins to deinitialize in parallel.
1111
while (reap_needed && (count= plugin_array.elements))
1114
for (idx= free_slots= 0; idx < count; idx++)
1116
plugin= *dynamic_element(&plugin_array, idx, struct st_plugin_int **);
1117
switch (plugin->state) {
1118
case PLUGIN_IS_READY:
1119
plugin->state= PLUGIN_IS_DELETED;
1122
case PLUGIN_IS_FREED:
1123
case PLUGIN_IS_UNINITIALIZED:
1131
release any plugin references held.
1133
unlock_variables(NULL, &global_system_variables);
1134
unlock_variables(NULL, &max_system_variables);
1138
if (count > free_slots)
1139
errmsg_printf(ERRMSG_LVL_WARN, _("Forcing shutdown of %"PRIu64" plugins"),
1140
(uint64_t)count - free_slots);
1142
plugins.reserve(count);
1145
If we have any plugins which did not die cleanly, we force shutdown
1147
for (idx= 0; idx < count; idx++)
1149
plugins.push_back(*dynamic_element(&plugin_array, idx,
1150
struct st_plugin_int **));
1151
/* change the state to ensure no reaping races */
1152
if (plugins[idx]->state == PLUGIN_IS_DELETED)
1153
plugins[idx]->state= PLUGIN_IS_DYING;
1157
We loop through all plugins and call deinit() if they have one.
1159
for (idx= 0; idx < count; idx++)
1160
if (!(plugins[idx]->state & (PLUGIN_IS_UNINITIALIZED | PLUGIN_IS_FREED)))
1162
errmsg_printf(ERRMSG_LVL_INFO, _("Plugin '%s' will be forced to shutdown"),
1163
plugins[idx]->name.str);
1165
We are forcing deinit on plugins so we don't want to do a ref_count
1166
check until we have processed all the plugins.
1168
plugin_deinitialize(plugins[idx], false);
1172
We defer checking ref_counts until after all plugins are deinitialized
1173
as some may have worker threads holding on to plugin references.
1175
for (idx= 0; idx < count; idx++)
1177
if (plugins[idx]->ref_count)
1178
errmsg_printf(ERRMSG_LVL_ERROR, _("Plugin '%s' has ref_count=%d after shutdown."),
1179
plugins[idx]->name.str, plugins[idx]->ref_count);
1180
if (plugins[idx]->state & PLUGIN_IS_UNINITIALIZED)
1181
plugin_del(plugins[idx]);
1185
Now we can deallocate all memory.
1188
cleanup_variables(NULL, &global_system_variables);
1189
cleanup_variables(NULL, &max_system_variables);
500
1195
/* Dispose of the memory */
501
plugin_mem_root.free_root(MYF(0));
1197
for (idx= 0; idx < DRIZZLE_MAX_PLUGIN_TYPE_NUM; idx++)
1198
hash_free(&plugin_hash[idx]);
1199
delete_dynamic(&plugin_array);
1201
count= plugin_dl_array.elements;
1203
for (idx= 0; idx < count; idx++)
1204
dl.push_back(*dynamic_element(&plugin_dl_array, idx,
1205
struct st_plugin_dl **));
1206
for (idx= 0; idx < count; idx++)
1207
free_plugin_mem(dl[idx]);
1208
delete_dynamic(&plugin_dl_array);
1210
hash_free(&bookmark_hash);
1211
free_root(&plugin_mem_root, MYF(0));
503
1213
global_variables_dynamic_size= 0;
1219
bool plugin_foreach_with_mask(Session *session, plugin_foreach_func *func,
1220
int type, uint32_t state_mask, void *arg)
1224
struct st_plugin_int *plugin;
1225
vector<st_plugin_int *> plugins;
1226
int version=plugin_array_version;
1231
state_mask= ~state_mask; // do it only once
1233
total= type == DRIZZLE_ANY_PLUGIN ? plugin_array.elements
1234
: plugin_hash[type].records;
1235
plugins.reserve(total);
1237
if (type == DRIZZLE_ANY_PLUGIN)
1239
for (idx= 0; idx < total; idx++)
1241
plugin= *dynamic_element(&plugin_array, idx, struct st_plugin_int **);
1242
plugins.push_back(!(plugin->state & state_mask) ? plugin : NULL);
1247
HASH *hash= plugin_hash + type;
1248
for (idx= 0; idx < total; idx++)
1250
plugin= (struct st_plugin_int *) hash_element(hash, idx);
1251
plugins.push_back(!(plugin->state & state_mask) ? plugin : NULL);
1254
for (idx= 0; idx < total; idx++)
1256
if (unlikely(version != plugin_array_version))
1258
for (uint32_t i=idx; i < total; i++)
1259
if (plugins[i] && plugins[i]->state & state_mask)
1262
plugin= plugins[idx];
1263
/* It will stop iterating on first engine error when "func" returns true */
1264
if (plugin && func(session, plugin_int_to_ref(plugin), arg))
1274
/****************************************************************************
1275
Internal type declarations for variables support
1276
****************************************************************************/
1278
#undef DRIZZLE_SYSVAR_NAME
1279
#define DRIZZLE_SYSVAR_NAME(name) name
1280
#define PLUGIN_VAR_TYPEMASK 0x007f
1282
#define EXTRA_OPTIONS 3 /* options for: 'foo', 'plugin-foo' and NULL */
1284
typedef DECLARE_DRIZZLE_SYSVAR_BASIC(sysvar_bool_t, bool);
1285
typedef DECLARE_DRIZZLE_SessionVAR_BASIC(sessionvar_bool_t, bool);
1286
typedef DECLARE_DRIZZLE_SYSVAR_BASIC(sysvar_str_t, char *);
1287
typedef DECLARE_DRIZZLE_SessionVAR_BASIC(sessionvar_str_t, char *);
1289
typedef DECLARE_DRIZZLE_SYSVAR_TYPELIB(sysvar_enum_t, unsigned long);
1290
typedef DECLARE_DRIZZLE_SessionVAR_TYPELIB(sessionvar_enum_t, unsigned long);
1291
typedef DECLARE_DRIZZLE_SYSVAR_TYPELIB(sysvar_set_t, uint64_t);
1292
typedef DECLARE_DRIZZLE_SessionVAR_TYPELIB(sessionvar_set_t, uint64_t);
1294
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_int_t, int);
1295
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_long_t, long);
1296
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_int64_t_t, int64_t);
1297
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_uint_t, uint);
1298
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_ulong_t, ulong);
1299
typedef DECLARE_DRIZZLE_SYSVAR_SIMPLE(sysvar_uint64_t_t, uint64_t);
1301
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_int_t, int);
1302
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_long_t, long);
1303
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_int64_t_t, int64_t);
1304
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_uint_t, uint);
1305
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_ulong_t, ulong);
1306
typedef DECLARE_DRIZZLE_SessionVAR_SIMPLE(sessionvar_uint64_t_t, uint64_t);
1308
typedef bool *(*mysql_sys_var_ptr_p)(Session* a_session, int offset);
1311
/****************************************************************************
1312
default variable data check and update functions
1313
****************************************************************************/
1315
static int check_func_bool(Session *, struct st_mysql_sys_var *var,
1316
void *save, st_mysql_value *value)
1318
char buff[STRING_BUFFER_USUAL_SIZE];
1319
const char *strvalue= "NULL", *str;
1323
if (value->value_type(value) == DRIZZLE_VALUE_TYPE_STRING)
1325
length= sizeof(buff);
1326
if (!(str= value->val_str(value, buff, &length)) ||
1327
(result= find_type(&bool_typelib, str, length, 1)-1) < 0)
1336
if (value->val_int(value, &tmp) < 0)
1346
*(int*)save= -result;
1349
my_error(ER_WRONG_VALUE_FOR_VAR, MYF(0), var->name, strvalue);
1354
static int check_func_int(Session *session, struct st_mysql_sys_var *var,
1355
void *save, st_mysql_value *value)
1359
struct my_option options;
1360
value->val_int(value, &tmp);
1361
plugin_opt_set_limits(&options, var);
1363
if (var->flags & PLUGIN_VAR_UNSIGNED)
1364
*(uint32_t *)save= (uint) getopt_ull_limit_value((uint64_t) tmp, &options,
1367
*(int *)save= (int) getopt_ll_limit_value(tmp, &options, &fixed);
1369
return throw_bounds_warning(session, fixed, var->flags & PLUGIN_VAR_UNSIGNED,
1370
var->name, (int64_t) tmp);
1374
static int check_func_long(Session *session, struct st_mysql_sys_var *var,
1375
void *save, st_mysql_value *value)
1379
struct my_option options;
1380
value->val_int(value, &tmp);
1381
plugin_opt_set_limits(&options, var);
1383
if (var->flags & PLUGIN_VAR_UNSIGNED)
1384
*(ulong *)save= (ulong) getopt_ull_limit_value((uint64_t) tmp, &options,
1387
*(long *)save= (long) getopt_ll_limit_value(tmp, &options, &fixed);
1389
return throw_bounds_warning(session, fixed, var->flags & PLUGIN_VAR_UNSIGNED,
1390
var->name, (int64_t) tmp);
1394
static int check_func_int64_t(Session *session, struct st_mysql_sys_var *var,
1395
void *save, st_mysql_value *value)
1399
struct my_option options;
1400
value->val_int(value, &tmp);
1401
plugin_opt_set_limits(&options, var);
1403
if (var->flags & PLUGIN_VAR_UNSIGNED)
1404
*(uint64_t *)save= getopt_ull_limit_value((uint64_t) tmp, &options,
1407
*(int64_t *)save= getopt_ll_limit_value(tmp, &options, &fixed);
1409
return throw_bounds_warning(session, fixed, var->flags & PLUGIN_VAR_UNSIGNED,
1410
var->name, (int64_t) tmp);
1413
static int check_func_str(Session *session, struct st_mysql_sys_var *,
1414
void *save, st_mysql_value *value)
1416
char buff[STRING_BUFFER_USUAL_SIZE];
1420
length= sizeof(buff);
1421
if ((str= value->val_str(value, buff, &length)))
1422
str= session->strmake(str, length);
1423
*(const char**)save= str;
1428
static int check_func_enum(Session *, struct st_mysql_sys_var *var,
1429
void *save, st_mysql_value *value)
1431
char buff[STRING_BUFFER_USUAL_SIZE];
1432
const char *strvalue= "NULL", *str;
1438
if (var->flags & PLUGIN_VAR_SessionLOCAL)
1439
typelib= ((sessionvar_enum_t*) var)->typelib;
1441
typelib= ((sysvar_enum_t*) var)->typelib;
1443
if (value->value_type(value) == DRIZZLE_VALUE_TYPE_STRING)
1445
length= sizeof(buff);
1446
if (!(str= value->val_str(value, buff, &length)))
1448
if ((result= (long)find_type(typelib, str, length, 1)-1) < 0)
1456
if (value->val_int(value, &tmp))
1458
if (tmp >= typelib->count)
1466
*(long*)save= result;
1469
my_error(ER_WRONG_VALUE_FOR_VAR, MYF(0), var->name, strvalue);
1474
static int check_func_set(Session *, struct st_mysql_sys_var *var,
1475
void *save, st_mysql_value *value)
1477
char buff[STRING_BUFFER_USUAL_SIZE], *error= 0;
1478
const char *strvalue= "NULL", *str;
1485
if (var->flags & PLUGIN_VAR_SessionLOCAL)
1486
typelib= ((sessionvar_set_t*) var)->typelib;
1488
typelib= ((sysvar_set_t*)var)->typelib;
1490
if (value->value_type(value) == DRIZZLE_VALUE_TYPE_STRING)
1492
length= sizeof(buff);
1493
if (!(str= value->val_str(value, buff, &length)))
1495
result= find_set(typelib, str, length, NULL,
1496
&error, &error_len, ¬_used);
1499
length= cmin(sizeof(buff), (unsigned long)error_len);
1500
strncpy(buff, error, length);
1508
if (value->val_int(value, (int64_t *)&result))
1510
if (unlikely((result >= (1UL << typelib->count)) &&
1511
(typelib->count < sizeof(long)*8)))
1513
llstr(result, buff);
1518
*(uint64_t*)save= result;
1521
my_error(ER_WRONG_VALUE_FOR_VAR, MYF(0), var->name, strvalue);
1526
static void update_func_bool(Session *, struct st_mysql_sys_var *,
1527
void *tgt, const void *save)
1529
*(bool *) tgt= *(int *) save ? 1 : 0;
1533
static void update_func_int(Session *, struct st_mysql_sys_var *,
1534
void *tgt, const void *save)
1536
*(int *)tgt= *(int *) save;
1540
static void update_func_long(Session *, struct st_mysql_sys_var *,
1541
void *tgt, const void *save)
1543
*(long *)tgt= *(long *) save;
1547
static void update_func_int64_t(Session *, struct st_mysql_sys_var *,
1548
void *tgt, const void *save)
1550
*(int64_t *)tgt= *(uint64_t *) save;
1554
static void update_func_str(Session *, struct st_mysql_sys_var *var,
1555
void *tgt, const void *save)
1557
char *old= *(char **) tgt;
1558
*(char **)tgt= *(char **) save;
1559
if (var->flags & PLUGIN_VAR_MEMALLOC)
1561
*(char **)tgt= strdup(*(char **) save);
1564
* There isn't a _really_ good thing to do here until this whole set_var
1565
* mess gets redesigned
1568
errmsg_printf(ERRMSG_LVL_ERROR, _("Out of memory."));
509
1576
****************************************************************************/
1579
sys_var *find_sys_var(Session *session, const char *str, uint32_t length)
1582
sys_var_pluginvar *pi= NULL;
1585
pthread_rwlock_rdlock(&LOCK_system_variables_hash);
1586
if ((var= intern_find_sys_var(str, length, false)) &&
1587
(pi= var->cast_pluginvar()))
1589
pthread_rwlock_unlock(&LOCK_system_variables_hash);
1590
LEX *lex= session ? session->lex : 0;
1591
if (!(plugin= my_intern_plugin_lock(lex, plugin_int_to_ref(pi->plugin))))
1592
var= NULL; /* failed to lock it, it must be uninstalling */
1594
if (!(plugin_state(plugin) & PLUGIN_IS_READY))
1596
/* initialization not completed */
1598
intern_plugin_unlock(lex, plugin);
1602
pthread_rwlock_unlock(&LOCK_system_variables_hash);
1605
If the variable exists but the plugin it is associated with is not ready
1606
then the intern_plugin_lock did not raise an error, so we do it here.
1609
my_error(ER_UNKNOWN_SYSTEM_VARIABLE, MYF(0), (char*) str);
1615
called by register_var, construct_options and test_plugin_options.
1616
Returns the 'bookmark' for the named variable.
1617
LOCK_system_variables_hash should be at least read locked
1619
static st_bookmark *find_bookmark(const char *plugin, const char *name, int flags)
1621
st_bookmark *result= NULL;
1622
uint32_t namelen, length, pluginlen= 0;
1625
if (!(flags & PLUGIN_VAR_SessionLOCAL))
1628
namelen= strlen(name);
1630
pluginlen= strlen(plugin) + 1;
1631
length= namelen + pluginlen + 2;
1632
varname= (char*) malloc(length);
1636
sprintf(varname+1,"%s_%s",plugin,name);
1637
for (p= varname + 1; *p; p++)
1642
memcpy(varname + 1, name, namelen + 1);
1644
varname[0]= flags & PLUGIN_VAR_TYPEMASK;
1646
result= (st_bookmark*) hash_search(&bookmark_hash,
1647
(const unsigned char*) varname, length - 1);
1655
returns a bookmark for session-local variables, creating if neccessary.
1656
returns null for non session-local variables.
1657
Requires that a write lock is obtained on LOCK_system_variables_hash
1659
static st_bookmark *register_var(const char *plugin, const char *name,
1662
uint32_t length= strlen(plugin) + strlen(name) + 3, size= 0, offset, new_size;
1663
st_bookmark *result;
1666
if (!(flags & PLUGIN_VAR_SessionLOCAL))
1669
switch (flags & PLUGIN_VAR_TYPEMASK) {
1670
case PLUGIN_VAR_BOOL:
1673
case PLUGIN_VAR_INT:
1676
case PLUGIN_VAR_LONG:
1677
case PLUGIN_VAR_ENUM:
1680
case PLUGIN_VAR_LONGLONG:
1681
case PLUGIN_VAR_SET:
1682
size= sizeof(uint64_t);
1684
case PLUGIN_VAR_STR:
1685
size= sizeof(char*);
1692
varname= ((char*) malloc(length));
1693
sprintf(varname+1, "%s_%s", plugin, name);
1694
for (p= varname + 1; *p; p++)
1698
if (!(result= find_bookmark(NULL, varname + 1, flags)))
1700
result= (st_bookmark*) alloc_root(&plugin_mem_root,
1701
sizeof(struct st_bookmark) + length-1);
1702
varname[0]= flags & PLUGIN_VAR_TYPEMASK;
1703
memcpy(result->key, varname, length);
1704
result->name_len= length - 2;
1707
assert(size && !(size & (size-1))); /* must be power of 2 */
1709
offset= global_system_variables.dynamic_variables_size;
1710
offset= (offset + size - 1) & ~(size - 1);
1711
result->offset= (int) offset;
1713
new_size= (offset + size + 63) & ~63;
1715
if (new_size > global_variables_dynamic_size)
1719
(char *)realloc(global_system_variables.dynamic_variables_ptr,
1722
global_system_variables.dynamic_variables_ptr= tmpptr;
1725
(char *)realloc(max_system_variables.dynamic_variables_ptr,
1728
max_system_variables.dynamic_variables_ptr= tmpptr;
1731
Clear the new variable value space. This is required for string
1732
variables. If their value is non-NULL, it must point to a valid
1735
memset(global_system_variables.dynamic_variables_ptr +
1736
global_variables_dynamic_size, 0,
1737
new_size - global_variables_dynamic_size);
1738
memset(max_system_variables.dynamic_variables_ptr +
1739
global_variables_dynamic_size, 0,
1740
new_size - global_variables_dynamic_size);
1741
global_variables_dynamic_size= new_size;
1744
global_system_variables.dynamic_variables_head= offset;
1745
max_system_variables.dynamic_variables_head= offset;
1746
global_system_variables.dynamic_variables_size= offset + size;
1747
max_system_variables.dynamic_variables_size= offset + size;
1748
global_system_variables.dynamic_variables_version++;
1749
max_system_variables.dynamic_variables_version++;
1751
result->version= global_system_variables.dynamic_variables_version;
1753
/* this should succeed because we have already checked if a dup exists */
1754
if (my_hash_insert(&bookmark_hash, (unsigned char*) result))
1756
fprintf(stderr, "failed to add placeholder to hash");
1766
returns a pointer to the memory which holds the session-local variable or
1767
a pointer to the global variable if session==null.
1768
If required, will sync with global variables if the requested variable
1769
has not yet been allocated in the current thread.
1771
static unsigned char *intern_sys_var_ptr(Session* session, int offset, bool global_lock)
1773
assert(offset >= 0);
1774
assert((uint)offset <= global_system_variables.dynamic_variables_head);
1777
return (unsigned char*) global_system_variables.dynamic_variables_ptr + offset;
1780
dynamic_variables_head points to the largest valid offset
1782
if (!session->variables.dynamic_variables_ptr ||
1783
(uint)offset > session->variables.dynamic_variables_head)
1787
pthread_rwlock_rdlock(&LOCK_system_variables_hash);
1790
if (!(tmpptr= (char *)realloc(session->variables.dynamic_variables_ptr,
1791
global_variables_dynamic_size)))
1793
session->variables.dynamic_variables_ptr= tmpptr;
1796
pthread_mutex_lock(&LOCK_global_system_variables);
1798
safe_mutex_assert_owner(&LOCK_global_system_variables);
1800
memcpy(session->variables.dynamic_variables_ptr +
1801
session->variables.dynamic_variables_size,
1802
global_system_variables.dynamic_variables_ptr +
1803
session->variables.dynamic_variables_size,
1804
global_system_variables.dynamic_variables_size -
1805
session->variables.dynamic_variables_size);
1808
now we need to iterate through any newly copied 'defaults'
1809
and if it is a string type with MEMALLOC flag, we need to strdup
1811
for (idx= 0; idx < bookmark_hash.records; idx++)
1813
sys_var_pluginvar *pi;
1815
st_bookmark *v= (st_bookmark*) hash_element(&bookmark_hash,idx);
1817
if (v->version <= session->variables.dynamic_variables_version ||
1818
!(var= intern_find_sys_var(v->key + 1, v->name_len, true)) ||
1819
!(pi= var->cast_pluginvar()) ||
1820
v->key[0] != (pi->plugin_var->flags & PLUGIN_VAR_TYPEMASK))
1823
/* Here we do anything special that may be required of the data types */
1825
if ((pi->plugin_var->flags & PLUGIN_VAR_TYPEMASK) == PLUGIN_VAR_STR &&
1826
pi->plugin_var->flags & PLUGIN_VAR_MEMALLOC)
1828
char **pp= (char**) (session->variables.dynamic_variables_ptr +
1829
*(int*)(pi->plugin_var + 1));
1830
if ((*pp= *(char**) (global_system_variables.dynamic_variables_ptr +
1831
*(int*)(pi->plugin_var + 1))))
1839
pthread_mutex_unlock(&LOCK_global_system_variables);
1841
session->variables.dynamic_variables_version=
1842
global_system_variables.dynamic_variables_version;
1843
session->variables.dynamic_variables_head=
1844
global_system_variables.dynamic_variables_head;
1845
session->variables.dynamic_variables_size=
1846
global_system_variables.dynamic_variables_size;
1848
pthread_rwlock_unlock(&LOCK_system_variables_hash);
1850
return (unsigned char*)session->variables.dynamic_variables_ptr + offset;
1853
static bool *mysql_sys_var_ptr_bool(Session* a_session, int offset)
1855
return (bool *)intern_sys_var_ptr(a_session, offset, true);
1858
static int *mysql_sys_var_ptr_int(Session* a_session, int offset)
1860
return (int *)intern_sys_var_ptr(a_session, offset, true);
1863
static long *mysql_sys_var_ptr_long(Session* a_session, int offset)
1865
return (long *)intern_sys_var_ptr(a_session, offset, true);
1868
static int64_t *mysql_sys_var_ptr_int64_t(Session* a_session, int offset)
1870
return (int64_t *)intern_sys_var_ptr(a_session, offset, true);
1873
static char **mysql_sys_var_ptr_str(Session* a_session, int offset)
1875
return (char **)intern_sys_var_ptr(a_session, offset, true);
1878
static uint64_t *mysql_sys_var_ptr_set(Session* a_session, int offset)
1880
return (uint64_t *)intern_sys_var_ptr(a_session, offset, true);
1883
static unsigned long *mysql_sys_var_ptr_enum(Session* a_session, int offset)
1885
return (unsigned long *)intern_sys_var_ptr(a_session, offset, true);
513
1889
void plugin_sessionvar_init(Session *session)
515
session->variables.storage_engine= NULL;
516
cleanup_variables(&session->variables);
1891
plugin_ref old_table_plugin= session->variables.table_plugin;
1893
session->variables.table_plugin= NULL;
1894
cleanup_variables(session, &session->variables);
518
1896
session->variables= global_system_variables;
519
session->variables.storage_engine= NULL;
1897
session->variables.table_plugin= NULL;
521
1899
/* we are going to allocate these lazily */
522
1900
session->variables.dynamic_variables_version= 0;
523
1901
session->variables.dynamic_variables_size= 0;
524
1902
session->variables.dynamic_variables_ptr= 0;
526
session->variables.storage_engine= global_system_variables.storage_engine;
1904
session->variables.table_plugin=
1905
my_intern_plugin_lock(NULL, global_system_variables.table_plugin);
1906
intern_plugin_unlock(NULL, old_table_plugin);
531
1912
Unlocks all system variables which hold a reference
533
static void unlock_variables(Session *, struct drizzle_system_variables *vars)
1914
static void unlock_variables(Session *, struct system_variables *vars)
535
vars->storage_engine= NULL;
1916
intern_plugin_unlock(NULL, vars->table_plugin);
1917
vars->table_plugin= NULL;
556
1966
void plugin_sessionvar_cleanup(Session *session)
558
1968
unlock_variables(session, &session->variables);
559
cleanup_variables(&session->variables);
1969
cleanup_variables(session, &session->variables);
1974
@brief Free values of thread variables of a plugin.
1976
This must be called before a plugin is deleted. Otherwise its
1977
variables are no longer accessible and the value space is lost. Note
1978
that only string values with PLUGIN_VAR_MEMALLOC are allocated and
1981
@param[in] vars Chain of system variables of a plugin
1984
static void plugin_vars_free_values(sys_var *vars)
1987
for (sys_var *var= vars; var; var= var->next)
1989
sys_var_pluginvar *piv= var->cast_pluginvar();
1991
((piv->plugin_var->flags & PLUGIN_VAR_TYPEMASK) == PLUGIN_VAR_STR) &&
1992
(piv->plugin_var->flags & PLUGIN_VAR_MEMALLOC))
1994
/* Free the string from global_system_variables. */
1995
char **valptr= (char**) piv->real_value_ptr(NULL, OPT_GLOBAL);
2004
bool sys_var_pluginvar::check_update_type(Item_result type)
2008
switch (plugin_var->flags & PLUGIN_VAR_TYPEMASK) {
2009
case PLUGIN_VAR_INT:
2010
case PLUGIN_VAR_LONG:
2011
case PLUGIN_VAR_LONGLONG:
2012
return type != INT_RESULT;
2013
case PLUGIN_VAR_STR:
2014
return type != STRING_RESULT;
2021
SHOW_TYPE sys_var_pluginvar::show_type()
2023
switch (plugin_var->flags & PLUGIN_VAR_TYPEMASK) {
2024
case PLUGIN_VAR_BOOL:
2025
return SHOW_MY_BOOL;
2026
case PLUGIN_VAR_INT:
2028
case PLUGIN_VAR_LONG:
2030
case PLUGIN_VAR_LONGLONG:
2031
return SHOW_LONGLONG;
2032
case PLUGIN_VAR_STR:
2033
return SHOW_CHAR_PTR;
2034
case PLUGIN_VAR_ENUM:
2035
case PLUGIN_VAR_SET:
2044
unsigned char* sys_var_pluginvar::real_value_ptr(Session *session, enum_var_type type)
2046
assert(session || (type == OPT_GLOBAL));
2047
if (plugin_var->flags & PLUGIN_VAR_SessionLOCAL)
2049
if (type == OPT_GLOBAL)
2052
return intern_sys_var_ptr(session, *(int*) (plugin_var+1), false);
2054
return *(unsigned char**) (plugin_var+1);
2058
TYPELIB* sys_var_pluginvar::plugin_var_typelib(void)
2060
switch (plugin_var->flags & (PLUGIN_VAR_TYPEMASK | PLUGIN_VAR_SessionLOCAL)) {
2061
case PLUGIN_VAR_ENUM:
2062
return ((sysvar_enum_t *)plugin_var)->typelib;
2063
case PLUGIN_VAR_SET:
2064
return ((sysvar_set_t *)plugin_var)->typelib;
2065
case PLUGIN_VAR_ENUM | PLUGIN_VAR_SessionLOCAL:
2066
return ((sessionvar_enum_t *)plugin_var)->typelib;
2067
case PLUGIN_VAR_SET | PLUGIN_VAR_SessionLOCAL:
2068
return ((sessionvar_set_t *)plugin_var)->typelib;
2076
unsigned char* sys_var_pluginvar::value_ptr(Session *session, enum_var_type type, LEX_STRING *)
2078
unsigned char* result;
2080
result= real_value_ptr(session, type);
2082
if ((plugin_var->flags & PLUGIN_VAR_TYPEMASK) == PLUGIN_VAR_ENUM)
2083
result= (unsigned char*) get_type(plugin_var_typelib(), *(ulong*)result);
2084
else if ((plugin_var->flags & PLUGIN_VAR_TYPEMASK) == PLUGIN_VAR_SET)
2086
char buffer[STRING_BUFFER_USUAL_SIZE];
2087
String str(buffer, sizeof(buffer), system_charset_info);
2088
TYPELIB *typelib= plugin_var_typelib();
2089
uint64_t mask= 1, value= *(uint64_t*) result;
2093
for (i= 0; i < typelib->count; i++, mask<<=1)
2095
if (!(value & mask))
2097
str.append(typelib->type_names[i], typelib->type_lengths[i]);
2101
result= (unsigned char*) "";
2103
result= (unsigned char*) session->strmake(str.ptr(), str.length()-1);
2109
bool sys_var_pluginvar::check(Session *session, set_var *var)
2111
st_item_value_holder value;
2112
assert(is_readonly() || plugin_var->check);
2114
value.value_type= item_value_type;
2115
value.val_str= item_val_str;
2116
value.val_int= item_val_int;
2117
value.val_real= item_val_real;
2118
value.item= var->value;
2120
return is_readonly() ||
2121
plugin_var->check(session, plugin_var, &var->save_result, &value);
2125
void sys_var_pluginvar::set_default(Session *session, enum_var_type type)
2130
assert(is_readonly() || plugin_var->update);
2135
pthread_mutex_lock(&LOCK_global_system_variables);
2136
tgt= real_value_ptr(session, type);
2137
src= ((void **) (plugin_var + 1) + 1);
2139
if (plugin_var->flags & PLUGIN_VAR_SessionLOCAL)
2141
if (type != OPT_GLOBAL)
2142
src= real_value_ptr(session, OPT_GLOBAL);
2144
switch (plugin_var->flags & PLUGIN_VAR_TYPEMASK) {
2145
case PLUGIN_VAR_INT:
2146
src= &((sessionvar_uint_t*) plugin_var)->def_val;
2148
case PLUGIN_VAR_LONG:
2149
src= &((sessionvar_ulong_t*) plugin_var)->def_val;
2151
case PLUGIN_VAR_LONGLONG:
2152
src= &((sessionvar_uint64_t_t*) plugin_var)->def_val;
2154
case PLUGIN_VAR_ENUM:
2155
src= &((sessionvar_enum_t*) plugin_var)->def_val;
2157
case PLUGIN_VAR_SET:
2158
src= &((sessionvar_set_t*) plugin_var)->def_val;
2160
case PLUGIN_VAR_BOOL:
2161
src= &((sessionvar_bool_t*) plugin_var)->def_val;
2163
case PLUGIN_VAR_STR:
2164
src= &((sessionvar_str_t*) plugin_var)->def_val;
2171
/* session must equal current_session if PLUGIN_VAR_SessionLOCAL flag is set */
2172
assert(!(plugin_var->flags & PLUGIN_VAR_SessionLOCAL) ||
2173
session == current_session);
2175
if (!(plugin_var->flags & PLUGIN_VAR_SessionLOCAL) || type == OPT_GLOBAL)
2177
plugin_var->update(session, plugin_var, tgt, src);
2178
pthread_mutex_unlock(&LOCK_global_system_variables);
2182
pthread_mutex_unlock(&LOCK_global_system_variables);
2183
plugin_var->update(session, plugin_var, tgt, src);
2188
bool sys_var_pluginvar::update(Session *session, set_var *var)
2192
assert(is_readonly() || plugin_var->update);
2194
/* session must equal current_session if PLUGIN_VAR_SessionLOCAL flag is set */
2195
assert(!(plugin_var->flags & PLUGIN_VAR_SessionLOCAL) ||
2196
session == current_session);
2201
pthread_mutex_lock(&LOCK_global_system_variables);
2202
tgt= real_value_ptr(session, var->type);
2204
if (!(plugin_var->flags & PLUGIN_VAR_SessionLOCAL) || var->type == OPT_GLOBAL)
2206
/* variable we are updating has global scope, so we unlock after updating */
2207
plugin_var->update(session, plugin_var, tgt, &var->save_result);
2208
pthread_mutex_unlock(&LOCK_global_system_variables);
2212
pthread_mutex_unlock(&LOCK_global_system_variables);
2213
plugin_var->update(session, plugin_var, tgt, &var->save_result);
2219
#define OPTION_SET_LIMITS(type, options, opt) \
2220
options->var_type= type; \
2221
options->def_value= (opt)->def_val; \
2222
options->min_value= (opt)->min_val; \
2223
options->max_value= (opt)->max_val; \
2224
options->block_size= (long) (opt)->blk_sz
2227
static void plugin_opt_set_limits(struct my_option *options,
2228
const struct st_mysql_sys_var *opt)
2230
options->sub_size= 0;
2232
switch (opt->flags & (PLUGIN_VAR_TYPEMASK |
2233
PLUGIN_VAR_UNSIGNED | PLUGIN_VAR_SessionLOCAL)) {
2234
/* global system variables */
2235
case PLUGIN_VAR_INT:
2236
OPTION_SET_LIMITS(GET_INT, options, (sysvar_int_t*) opt);
2238
case PLUGIN_VAR_INT | PLUGIN_VAR_UNSIGNED:
2239
OPTION_SET_LIMITS(GET_UINT, options, (sysvar_uint_t*) opt);
2241
case PLUGIN_VAR_LONG:
2242
OPTION_SET_LIMITS(GET_LONG, options, (sysvar_long_t*) opt);
2244
case PLUGIN_VAR_LONG | PLUGIN_VAR_UNSIGNED:
2245
OPTION_SET_LIMITS(GET_ULONG, options, (sysvar_ulong_t*) opt);
2247
case PLUGIN_VAR_LONGLONG:
2248
OPTION_SET_LIMITS(GET_LL, options, (sysvar_int64_t_t*) opt);
2250
case PLUGIN_VAR_LONGLONG | PLUGIN_VAR_UNSIGNED:
2251
OPTION_SET_LIMITS(GET_ULL, options, (sysvar_uint64_t_t*) opt);
2253
case PLUGIN_VAR_ENUM:
2254
options->var_type= GET_ENUM;
2255
options->typelib= ((sysvar_enum_t*) opt)->typelib;
2256
options->def_value= ((sysvar_enum_t*) opt)->def_val;
2257
options->min_value= options->block_size= 0;
2258
options->max_value= options->typelib->count - 1;
2260
case PLUGIN_VAR_SET:
2261
options->var_type= GET_SET;
2262
options->typelib= ((sysvar_set_t*) opt)->typelib;
2263
options->def_value= ((sysvar_set_t*) opt)->def_val;
2264
options->min_value= options->block_size= 0;
2265
options->max_value= (1UL << options->typelib->count) - 1;
2267
case PLUGIN_VAR_BOOL:
2268
options->var_type= GET_BOOL;
2269
options->def_value= ((sysvar_bool_t*) opt)->def_val;
2271
case PLUGIN_VAR_STR:
2272
options->var_type= ((opt->flags & PLUGIN_VAR_MEMALLOC) ?
2273
GET_STR_ALLOC : GET_STR);
2274
options->def_value= (intptr_t) ((sysvar_str_t*) opt)->def_val;
2276
/* threadlocal variables */
2277
case PLUGIN_VAR_INT | PLUGIN_VAR_SessionLOCAL:
2278
OPTION_SET_LIMITS(GET_INT, options, (sessionvar_int_t*) opt);
2280
case PLUGIN_VAR_INT | PLUGIN_VAR_UNSIGNED | PLUGIN_VAR_SessionLOCAL:
2281
OPTION_SET_LIMITS(GET_UINT, options, (sessionvar_uint_t*) opt);
2283
case PLUGIN_VAR_LONG | PLUGIN_VAR_SessionLOCAL:
2284
OPTION_SET_LIMITS(GET_LONG, options, (sessionvar_long_t*) opt);
2286
case PLUGIN_VAR_LONG | PLUGIN_VAR_UNSIGNED | PLUGIN_VAR_SessionLOCAL:
2287
OPTION_SET_LIMITS(GET_ULONG, options, (sessionvar_ulong_t*) opt);
2289
case PLUGIN_VAR_LONGLONG | PLUGIN_VAR_SessionLOCAL:
2290
OPTION_SET_LIMITS(GET_LL, options, (sessionvar_int64_t_t*) opt);
2292
case PLUGIN_VAR_LONGLONG | PLUGIN_VAR_UNSIGNED | PLUGIN_VAR_SessionLOCAL:
2293
OPTION_SET_LIMITS(GET_ULL, options, (sessionvar_uint64_t_t*) opt);
2295
case PLUGIN_VAR_ENUM | PLUGIN_VAR_SessionLOCAL:
2296
options->var_type= GET_ENUM;
2297
options->typelib= ((sessionvar_enum_t*) opt)->typelib;
2298
options->def_value= ((sessionvar_enum_t*) opt)->def_val;
2299
options->min_value= options->block_size= 0;
2300
options->max_value= options->typelib->count - 1;
2302
case PLUGIN_VAR_SET | PLUGIN_VAR_SessionLOCAL:
2303
options->var_type= GET_SET;
2304
options->typelib= ((sessionvar_set_t*) opt)->typelib;
2305
options->def_value= ((sessionvar_set_t*) opt)->def_val;
2306
options->min_value= options->block_size= 0;
2307
options->max_value= (1UL << options->typelib->count) - 1;
2309
case PLUGIN_VAR_BOOL | PLUGIN_VAR_SessionLOCAL:
2310
options->var_type= GET_BOOL;
2311
options->def_value= ((sessionvar_bool_t*) opt)->def_val;
2313
case PLUGIN_VAR_STR | PLUGIN_VAR_SessionLOCAL:
2314
options->var_type= ((opt->flags & PLUGIN_VAR_MEMALLOC) ?
2315
GET_STR_ALLOC : GET_STR);
2316
options->def_value= (intptr_t) ((sessionvar_str_t*) opt)->def_val;
2321
options->arg_type= REQUIRED_ARG;
2322
if (opt->flags & PLUGIN_VAR_NOCMDARG)
2323
options->arg_type= NO_ARG;
2324
if (opt->flags & PLUGIN_VAR_OPCMDARG)
2325
options->arg_type= OPT_ARG;
2328
extern "C" bool get_one_plugin_option(int optid, const struct my_option *,
2331
bool get_one_plugin_option(int, const struct my_option *, char *)
2337
static int construct_options(MEM_ROOT *mem_root, struct st_plugin_int *tmp,
2338
my_option *options, bool can_disable)
2340
const char *plugin_name= tmp->plugin->name;
2341
uint32_t namelen= strlen(plugin_name), optnamelen;
2342
uint32_t buffer_length= namelen * 4 + (can_disable ? 75 : 10);
2343
char *name= (char*) alloc_root(mem_root, buffer_length) + 1;
2345
int index= 0, offset= 0;
2346
st_mysql_sys_var *opt, **plugin_option;
2349
/* support --skip-plugin-foo syntax */
2350
memcpy(name, plugin_name, namelen + 1);
2351
my_casedn_str(&my_charset_utf8_general_ci, name);
2352
sprintf(name+namelen+1, "plugin-%s", name);
2353
/* Now we have namelen + 1 + 7 + namelen + 1 == namelen * 2 + 9. */
2355
for (p= name + namelen*2 + 8; p > name; p--)
2361
sprintf(name+namelen*2+10,
2362
"Enable %s plugin. Disable with --skip-%s (will save memory).",
2365
Now we have namelen * 2 + 10 (one char unused) + 7 + namelen + 9 +
2366
20 + namelen + 20 + 1 == namelen * 4 + 67.
2369
options[0].comment= name + namelen*2 + 10;
2372
options[1].name= (options[0].name= name) + namelen + 1;
2373
options[0].id= options[1].id= 256; /* must be >255. dup id ok */
2374
options[0].var_type= options[1].var_type= GET_BOOL;
2375
options[0].arg_type= options[1].arg_type= NO_ARG;
2376
options[0].def_value= options[1].def_value= true;
2377
options[0].value= options[0].u_max_value=
2378
options[1].value= options[1].u_max_value= (char**) (name - 1);
2382
Two passes as the 2nd pass will take pointer addresses for use
2383
by my_getopt and register_var() in the first pass uses realloc
2386
for (plugin_option= tmp->plugin->system_vars;
2387
plugin_option && *plugin_option; plugin_option++, index++)
2389
opt= *plugin_option;
2390
if (!(opt->flags & PLUGIN_VAR_SessionLOCAL))
2392
if (!(register_var(name, opt->name, opt->flags)))
2394
switch (opt->flags & PLUGIN_VAR_TYPEMASK) {
2395
case PLUGIN_VAR_BOOL:
2396
(((sessionvar_bool_t *)opt)->resolve)= mysql_sys_var_ptr_bool;
2398
case PLUGIN_VAR_INT:
2399
(((sessionvar_int_t *)opt)->resolve)= mysql_sys_var_ptr_int;
2401
case PLUGIN_VAR_LONG:
2402
(((sessionvar_long_t *)opt)->resolve)= mysql_sys_var_ptr_long;
2404
case PLUGIN_VAR_LONGLONG:
2405
(((sessionvar_int64_t_t *)opt)->resolve)= mysql_sys_var_ptr_int64_t;
2407
case PLUGIN_VAR_STR:
2408
(((sessionvar_str_t *)opt)->resolve)= mysql_sys_var_ptr_str;
2410
case PLUGIN_VAR_ENUM:
2411
(((sessionvar_enum_t *)opt)->resolve)= mysql_sys_var_ptr_enum;
2413
case PLUGIN_VAR_SET:
2414
(((sessionvar_set_t *)opt)->resolve)= mysql_sys_var_ptr_set;
2417
errmsg_printf(ERRMSG_LVL_ERROR, _("Unknown variable type code 0x%x in plugin '%s'."),
2418
opt->flags, plugin_name);
2423
for (plugin_option= tmp->plugin->system_vars;
2424
plugin_option && *plugin_option; plugin_option++, index++)
2426
switch ((opt= *plugin_option)->flags & PLUGIN_VAR_TYPEMASK) {
2427
case PLUGIN_VAR_BOOL:
2429
opt->check= check_func_bool;
2431
opt->update= update_func_bool;
2433
case PLUGIN_VAR_INT:
2435
opt->check= check_func_int;
2437
opt->update= update_func_int;
2439
case PLUGIN_VAR_LONG:
2441
opt->check= check_func_long;
2443
opt->update= update_func_long;
2445
case PLUGIN_VAR_LONGLONG:
2447
opt->check= check_func_int64_t;
2449
opt->update= update_func_int64_t;
2451
case PLUGIN_VAR_STR:
2453
opt->check= check_func_str;
2456
opt->update= update_func_str;
2457
if ((opt->flags & (PLUGIN_VAR_MEMALLOC | PLUGIN_VAR_READONLY)) == false)
2459
opt->flags|= PLUGIN_VAR_READONLY;
2460
errmsg_printf(ERRMSG_LVL_WARN, _("Server variable %s of plugin %s was forced "
2461
"to be read-only: string variable without "
2462
"update_func and PLUGIN_VAR_MEMALLOC flag"),
2463
opt->name, plugin_name);
2467
case PLUGIN_VAR_ENUM:
2469
opt->check= check_func_enum;
2471
opt->update= update_func_long;
2473
case PLUGIN_VAR_SET:
2475
opt->check= check_func_set;
2477
opt->update= update_func_int64_t;
2480
errmsg_printf(ERRMSG_LVL_ERROR, _("Unknown variable type code 0x%x in plugin '%s'."),
2481
opt->flags, plugin_name);
2485
if ((opt->flags & (PLUGIN_VAR_NOCMDOPT | PLUGIN_VAR_SessionLOCAL))
2486
== PLUGIN_VAR_NOCMDOPT)
2491
errmsg_printf(ERRMSG_LVL_ERROR, _("Missing variable name in plugin '%s'."),
2496
if (!(opt->flags & PLUGIN_VAR_SessionLOCAL))
2498
optnamelen= strlen(opt->name);
2499
optname= (char*) alloc_root(mem_root, namelen + optnamelen + 2);
2500
sprintf(optname, "%s-%s", name, opt->name);
2501
optnamelen= namelen + optnamelen + 1;
2505
/* this should not fail because register_var should create entry */
2506
if (!(v= find_bookmark(name, opt->name, opt->flags)))
2508
errmsg_printf(ERRMSG_LVL_ERROR, _("Thread local variable '%s' not allocated "
2509
"in plugin '%s'."), opt->name, plugin_name);
2513
*(int*)(opt + 1)= offset= v->offset;
2515
if (opt->flags & PLUGIN_VAR_NOCMDOPT)
2518
optname= (char*) memdup_root(mem_root, v->key + 1,
2519
(optnamelen= v->name_len) + 1);
2522
/* convert '_' to '-' */
2523
for (p= optname; *p; p++)
2527
options->name= optname;
2528
options->comment= opt->comment;
2529
options->app_type= opt;
2530
options->id= (options-1)->id + 1;
2532
plugin_opt_set_limits(options, opt);
2534
if (opt->flags & PLUGIN_VAR_SessionLOCAL)
2535
options->value= options->u_max_value= (char**)
2536
(global_system_variables.dynamic_variables_ptr + offset);
2538
options->value= options->u_max_value= *(char***) (opt + 1);
2540
options[1]= options[0];
2541
options[1].name= p= (char*) alloc_root(mem_root, optnamelen + 8);
2542
options[1].comment= 0; // hidden
2543
sprintf(p,"plugin-%s",optname);
2552
static my_option *construct_help_options(MEM_ROOT *mem_root,
2553
struct st_plugin_int *p)
2555
st_mysql_sys_var **opt;
2558
uint32_t count= EXTRA_OPTIONS;
2560
for (opt= p->plugin->system_vars; opt && *opt; opt++, count+= 2) {};
2562
if (!(opts= (my_option*) alloc_root(mem_root, sizeof(my_option) * count)))
2565
memset(opts, 0, sizeof(my_option) * count);
2567
if ((my_strcasecmp(&my_charset_utf8_general_ci, p->name.str, "MyISAM") == 0))
2569
else if ((my_strcasecmp(&my_charset_utf8_general_ci, p->name.str, "MEMORY") == 0))
2575
if (construct_options(mem_root, p, opts, can_disable))