mirror of
https://github.com/VCMP-SqMod/SqMod.git
synced 2025-01-19 12:07:13 +01:00
509 lines
13 KiB
C
509 lines
13 KiB
C
/* Copyright (C) 2010 - 2012 Sergei Golubchik and Monty Program Ab
|
|
2015-2016 MariaDB Corporation AB
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Library General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2 of the License, or (at your option) any later version.
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Library General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Library General Public
|
|
License along with this library; if not see <http://www.gnu.org/licenses>
|
|
or write to the Free Software Foundation, Inc.,
|
|
51 Franklin St., Fifth Floor, Boston, MA 02110, USA */
|
|
|
|
/**
|
|
@file
|
|
|
|
Support code for the client side (libmariadb) plugins
|
|
|
|
Client plugins are somewhat different from server plugins, they are simpler.
|
|
|
|
They do not need to be installed or in any way explicitly loaded on the
|
|
client, they are loaded automatically on demand.
|
|
One client plugin per shared object, soname *must* match the plugin name.
|
|
|
|
There is no reference counting and no unloading either.
|
|
*/
|
|
|
|
/* Silence warnings about variable 'unused' being used. */
|
|
#define FORCE_INIT_OF_VARS 1
|
|
|
|
#include <ma_global.h>
|
|
#include <ma_sys.h>
|
|
#include <ma_common.h>
|
|
#include <ma_string.h>
|
|
#include <ma_pthread.h>
|
|
|
|
#include "errmsg.h"
|
|
#include <mysql/client_plugin.h>
|
|
|
|
#ifndef WIN32
|
|
#include <dlfcn.h>
|
|
#endif
|
|
|
|
struct st_client_plugin_int {
|
|
struct st_client_plugin_int *next;
|
|
void *dlhandle;
|
|
struct st_mysql_client_plugin *plugin;
|
|
};
|
|
|
|
static my_bool initialized= 0;
|
|
static MA_MEM_ROOT mem_root;
|
|
|
|
static uint valid_plugins[][2]= {
|
|
{MYSQL_CLIENT_AUTHENTICATION_PLUGIN, MYSQL_CLIENT_AUTHENTICATION_PLUGIN_INTERFACE_VERSION},
|
|
{MARIADB_CLIENT_PVIO_PLUGIN, MARIADB_CLIENT_PVIO_PLUGIN_INTERFACE_VERSION},
|
|
{MARIADB_CLIENT_TRACE_PLUGIN, MARIADB_CLIENT_TRACE_PLUGIN_INTERFACE_VERSION},
|
|
{MARIADB_CLIENT_REMOTEIO_PLUGIN, MARIADB_CLIENT_REMOTEIO_PLUGIN_INTERFACE_VERSION},
|
|
{MARIADB_CLIENT_CONNECTION_PLUGIN, MARIADB_CLIENT_CONNECTION_PLUGIN_INTERFACE_VERSION},
|
|
{0, 0}
|
|
};
|
|
|
|
/*
|
|
Loaded plugins are stored in a linked list.
|
|
The list is append-only, the elements are added to the head (like in a stack).
|
|
The elements are added under a mutex, but the list can be read and traversed
|
|
without any mutex because once an element is added to the list, it stays
|
|
there. The main purpose of a mutex is to prevent two threads from
|
|
loading the same plugin twice in parallel.
|
|
*/
|
|
|
|
|
|
struct st_client_plugin_int *plugin_list[MYSQL_CLIENT_MAX_PLUGINS + MARIADB_CLIENT_MAX_PLUGINS];
|
|
#ifdef THREAD
|
|
static pthread_mutex_t LOCK_load_client_plugin;
|
|
#endif
|
|
|
|
@EXTERNAL_PLUGINS@
|
|
|
|
struct st_mysql_client_plugin *mysql_client_builtins[]=
|
|
{
|
|
@BUILTIN_PLUGINS@
|
|
0
|
|
};
|
|
|
|
|
|
static int is_not_initialized(MYSQL *mysql, const char *name)
|
|
{
|
|
if (initialized)
|
|
return 0;
|
|
|
|
my_set_error(mysql, CR_AUTH_PLUGIN_CANNOT_LOAD,
|
|
SQLSTATE_UNKNOWN, ER(CR_AUTH_PLUGIN_CANNOT_LOAD),
|
|
name, "not initialized");
|
|
return 1;
|
|
}
|
|
|
|
static int get_plugin_nr(uint type)
|
|
{
|
|
uint i= 0;
|
|
for(; valid_plugins[i][1]; i++)
|
|
if (valid_plugins[i][0] == type)
|
|
return i;
|
|
return -1;
|
|
}
|
|
|
|
static const char *check_plugin_version(struct st_mysql_client_plugin *plugin, unsigned int version)
|
|
{
|
|
if (plugin->interface_version < version ||
|
|
(plugin->interface_version >> 8) > (version >> 8))
|
|
return "Incompatible client plugin interface";
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
finds a plugin in the list
|
|
|
|
@param name plugin name to search for
|
|
@param type plugin type
|
|
|
|
@note this does NOT necessarily need a mutex, take care!
|
|
|
|
@retval a pointer to a found plugin or 0
|
|
*/
|
|
static struct st_mysql_client_plugin *find_plugin(const char *name, int type)
|
|
{
|
|
struct st_client_plugin_int *p;
|
|
int plugin_nr= get_plugin_nr(type);
|
|
|
|
DBUG_ASSERT(initialized);
|
|
if (plugin_nr == -1)
|
|
return 0;
|
|
|
|
if (!name)
|
|
return plugin_list[plugin_nr]->plugin;
|
|
|
|
for (p= plugin_list[plugin_nr]; p; p= p->next)
|
|
{
|
|
if (strcmp(p->plugin->name, name) == 0)
|
|
return p->plugin;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/**
|
|
verifies the plugin and adds it to the list
|
|
|
|
@param mysql MYSQL structure (for error reporting)
|
|
@param plugin plugin to install
|
|
@param dlhandle a handle to the shared object (returned by dlopen)
|
|
or 0 if the plugin was not dynamically loaded
|
|
@param argc number of arguments in the 'va_list args'
|
|
@param args arguments passed to the plugin initialization function
|
|
|
|
@retval a pointer to an installed plugin or 0
|
|
*/
|
|
|
|
static struct st_mysql_client_plugin *
|
|
add_plugin(MYSQL *mysql, struct st_mysql_client_plugin *plugin, void *dlhandle,
|
|
int argc, va_list args)
|
|
{
|
|
const char *errmsg;
|
|
struct st_client_plugin_int plugin_int, *p;
|
|
char errbuf[1024];
|
|
int plugin_nr;
|
|
|
|
DBUG_ASSERT(initialized);
|
|
|
|
plugin_int.plugin= plugin;
|
|
plugin_int.dlhandle= dlhandle;
|
|
|
|
if ((plugin_nr= get_plugin_nr(plugin->type)) == -1)
|
|
{
|
|
errmsg= "Unknown client plugin type";
|
|
goto err1;
|
|
}
|
|
if ((errmsg= check_plugin_version(plugin, valid_plugins[plugin_nr][1])))
|
|
goto err1;
|
|
|
|
/* Call the plugin initialization function, if any */
|
|
if (plugin->init && plugin->init(errbuf, sizeof(errbuf), argc, args))
|
|
{
|
|
errmsg= errbuf;
|
|
goto err1;
|
|
}
|
|
|
|
p= (struct st_client_plugin_int *)
|
|
ma_memdup_root(&mem_root, (char *)&plugin_int, sizeof(plugin_int));
|
|
|
|
if (!p)
|
|
{
|
|
errmsg= "Out of memory";
|
|
goto err2;
|
|
}
|
|
|
|
|
|
p->next= plugin_list[plugin_nr];
|
|
plugin_list[plugin_nr]= p;
|
|
|
|
return plugin;
|
|
|
|
err2:
|
|
if (plugin->deinit)
|
|
plugin->deinit();
|
|
err1:
|
|
my_set_error(mysql, CR_AUTH_PLUGIN_CANNOT_LOAD, SQLSTATE_UNKNOWN,
|
|
ER(CR_AUTH_PLUGIN_CANNOT_LOAD), plugin->name, errmsg);
|
|
if (dlhandle)
|
|
(void)dlclose(dlhandle);
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/**
|
|
Loads plugins which are specified in the environment variable
|
|
LIBMYSQL_PLUGINS.
|
|
|
|
Multiple plugins must be separated by semicolon. This function doesn't
|
|
return or log an error.
|
|
|
|
The function is be called by mysql_client_plugin_init
|
|
|
|
@todo
|
|
Support extended syntax, passing parameters to plugins, for example
|
|
LIBMYSQL_PLUGINS="plugin1(param1,param2);plugin2;..."
|
|
or
|
|
LIBMYSQL_PLUGINS="plugin1=int:param1,str:param2;plugin2;..."
|
|
*/
|
|
|
|
static void load_env_plugins(MYSQL *mysql)
|
|
{
|
|
char *plugs, *free_env, *s= getenv("LIBMYSQL_PLUGINS");
|
|
|
|
if (ma_check_env_str(s))
|
|
return;
|
|
|
|
free_env= strdup(s);
|
|
plugs= s= free_env;
|
|
|
|
do {
|
|
if ((s= strchr(plugs, ';')))
|
|
*s= '\0';
|
|
mysql_load_plugin(mysql, plugs, -1, 0);
|
|
plugs= s + 1;
|
|
} while (s);
|
|
|
|
free(free_env);
|
|
}
|
|
|
|
/********** extern functions to be used by libmariadb *********************/
|
|
|
|
/**
|
|
Initializes the client plugin layer.
|
|
|
|
This function must be called before any other client plugin function.
|
|
|
|
@retval 0 successful
|
|
@retval != 0 error occurred
|
|
*/
|
|
|
|
int mysql_client_plugin_init()
|
|
{
|
|
MYSQL mysql;
|
|
struct st_mysql_client_plugin **builtin;
|
|
va_list unused;
|
|
LINT_INIT_STRUCT(unused);
|
|
|
|
if (initialized)
|
|
return 0;
|
|
|
|
memset(&mysql, 0, sizeof(mysql)); /* dummy mysql for set_mysql_extended_error */
|
|
|
|
pthread_mutex_init(&LOCK_load_client_plugin, NULL);
|
|
ma_init_alloc_root(&mem_root, 128, 128);
|
|
|
|
memset(&plugin_list, 0, sizeof(plugin_list));
|
|
|
|
initialized= 1;
|
|
|
|
pthread_mutex_lock(&LOCK_load_client_plugin);
|
|
for (builtin= mysql_client_builtins; *builtin; builtin++)
|
|
add_plugin(&mysql, *builtin, 0, 0, unused);
|
|
|
|
pthread_mutex_unlock(&LOCK_load_client_plugin);
|
|
|
|
load_env_plugins(&mysql);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/**
|
|
Deinitializes the client plugin layer.
|
|
|
|
Unloades all client plugins and frees any associated resources.
|
|
*/
|
|
|
|
void mysql_client_plugin_deinit()
|
|
{
|
|
int i;
|
|
struct st_client_plugin_int *p;
|
|
|
|
if (!initialized)
|
|
return;
|
|
|
|
for (i=0; i < MYSQL_CLIENT_MAX_PLUGINS; i++)
|
|
for (p= plugin_list[i]; p; p= p->next)
|
|
{
|
|
if (p->plugin->deinit)
|
|
p->plugin->deinit();
|
|
if (p->dlhandle)
|
|
(void)dlclose(p->dlhandle);
|
|
}
|
|
|
|
memset(&plugin_list, 0, sizeof(plugin_list));
|
|
initialized= 0;
|
|
ma_free_root(&mem_root, MYF(0));
|
|
pthread_mutex_destroy(&LOCK_load_client_plugin);
|
|
}
|
|
|
|
/************* public facing functions, for client consumption *********/
|
|
|
|
/* see <mysql/client_plugin.h> for a full description */
|
|
struct st_mysql_client_plugin * STDCALL
|
|
mysql_client_register_plugin(MYSQL *mysql,
|
|
struct st_mysql_client_plugin *plugin)
|
|
{
|
|
va_list unused;
|
|
LINT_INIT_STRUCT(unused);
|
|
|
|
if (is_not_initialized(mysql, plugin->name))
|
|
return NULL;
|
|
|
|
pthread_mutex_lock(&LOCK_load_client_plugin);
|
|
|
|
/* make sure the plugin wasn't loaded meanwhile */
|
|
if (find_plugin(plugin->name, plugin->type))
|
|
{
|
|
my_set_error(mysql, CR_AUTH_PLUGIN_CANNOT_LOAD,
|
|
SQLSTATE_UNKNOWN, ER(CR_AUTH_PLUGIN_CANNOT_LOAD),
|
|
plugin->name, "it is already loaded");
|
|
plugin= NULL;
|
|
}
|
|
else
|
|
plugin= add_plugin(mysql, plugin, 0, 0, unused);
|
|
|
|
pthread_mutex_unlock(&LOCK_load_client_plugin);
|
|
return plugin;
|
|
}
|
|
|
|
|
|
/* see <mysql/client_plugin.h> for a full description */
|
|
struct st_mysql_client_plugin * STDCALL
|
|
mysql_load_plugin_v(MYSQL *mysql, const char *name, int type,
|
|
int argc, va_list args)
|
|
{
|
|
const char *errmsg;
|
|
#ifdef _WIN32
|
|
char errbuf[1024];
|
|
#endif
|
|
char dlpath[FN_REFLEN+1];
|
|
void *sym, *dlhandle = NULL;
|
|
struct st_mysql_client_plugin *plugin;
|
|
char *env_plugin_dir= getenv("MARIADB_PLUGIN_DIR");
|
|
|
|
CLEAR_CLIENT_ERROR(mysql);
|
|
if (is_not_initialized(mysql, name))
|
|
return NULL;
|
|
|
|
pthread_mutex_lock(&LOCK_load_client_plugin);
|
|
|
|
/* make sure the plugin wasn't loaded meanwhile */
|
|
if (type >= 0 && find_plugin(name, type))
|
|
{
|
|
errmsg= "it is already loaded";
|
|
goto err;
|
|
}
|
|
|
|
/* Compile dll path */
|
|
#ifndef WIN32
|
|
snprintf(dlpath, sizeof(dlpath) - 1, "%s/%s%s",
|
|
mysql->options.extension && mysql->options.extension->plugin_dir ?
|
|
mysql->options.extension->plugin_dir : (env_plugin_dir) ? env_plugin_dir :
|
|
MARIADB_PLUGINDIR, name, SO_EXT);
|
|
#else
|
|
{
|
|
char *p= (mysql->options.extension && mysql->options.extension->plugin_dir) ?
|
|
mysql->options.extension->plugin_dir : env_plugin_dir;
|
|
snprintf(dlpath, sizeof(dlpath), "%s%s%s%s", p ? p : "", p ? "\\" : "", name, SO_EXT);
|
|
}
|
|
#endif
|
|
|
|
if (strpbrk(name, "()[]!@#$%^&/*;.,'?\\"))
|
|
{
|
|
errmsg= "invalid plugin name";
|
|
goto err;
|
|
}
|
|
|
|
|
|
/* Open new dll handle */
|
|
if (!(dlhandle= dlopen((const char *)dlpath, RTLD_NOW)))
|
|
{
|
|
#ifdef _WIN32
|
|
char winmsg[255];
|
|
size_t len;
|
|
winmsg[0] = 0;
|
|
FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM,
|
|
NULL,
|
|
GetLastError(),
|
|
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
|
|
winmsg, 255, NULL);
|
|
len= strlen(winmsg);
|
|
while (len > 0 && (winmsg[len - 1] == '\n' || winmsg[len - 1] == '\r'))
|
|
len--;
|
|
if (len)
|
|
winmsg[len] = 0;
|
|
snprintf(errbuf, sizeof(errbuf), "%s Library path is '%s'", winmsg, dlpath);
|
|
errmsg= errbuf;
|
|
#else
|
|
errmsg= dlerror();
|
|
#endif
|
|
goto err;
|
|
}
|
|
|
|
|
|
if (!(sym= dlsym(dlhandle, plugin_declarations_sym)))
|
|
{
|
|
errmsg= "not a plugin";
|
|
(void)dlclose(dlhandle);
|
|
goto err;
|
|
}
|
|
|
|
plugin= (struct st_mysql_client_plugin*)sym;
|
|
|
|
if (type >=0 && type != plugin->type)
|
|
{
|
|
errmsg= "type mismatch";
|
|
goto err;
|
|
}
|
|
|
|
if (strcmp(name, plugin->name))
|
|
{
|
|
errmsg= "name mismatch";
|
|
goto err;
|
|
}
|
|
|
|
if (type < 0 && find_plugin(name, plugin->type))
|
|
{
|
|
errmsg= "it is already loaded";
|
|
goto err;
|
|
}
|
|
|
|
plugin= add_plugin(mysql, plugin, dlhandle, argc, args);
|
|
|
|
pthread_mutex_unlock(&LOCK_load_client_plugin);
|
|
|
|
return plugin;
|
|
|
|
err:
|
|
if (dlhandle)
|
|
dlclose(dlhandle);
|
|
pthread_mutex_unlock(&LOCK_load_client_plugin);
|
|
my_set_error(mysql, CR_AUTH_PLUGIN_CANNOT_LOAD, SQLSTATE_UNKNOWN,
|
|
ER(CR_AUTH_PLUGIN_CANNOT_LOAD), name, errmsg);
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/* see <mysql/client_plugin.h> for a full description */
|
|
struct st_mysql_client_plugin * STDCALL
|
|
mysql_load_plugin(MYSQL *mysql, const char *name, int type, int argc, ...)
|
|
{
|
|
struct st_mysql_client_plugin *p;
|
|
va_list args;
|
|
va_start(args, argc);
|
|
p= mysql_load_plugin_v(mysql, name, type, argc, args);
|
|
va_end(args);
|
|
return p;
|
|
}
|
|
|
|
/* see <mysql/client_plugin.h> for a full description */
|
|
struct st_mysql_client_plugin * STDCALL
|
|
mysql_client_find_plugin(MYSQL *mysql, const char *name, int type)
|
|
{
|
|
struct st_mysql_client_plugin *p;
|
|
int plugin_nr= get_plugin_nr(type);
|
|
|
|
if (is_not_initialized(mysql, name))
|
|
return NULL;
|
|
|
|
if (plugin_nr == -1)
|
|
{
|
|
my_set_error(mysql, CR_AUTH_PLUGIN_CANNOT_LOAD, SQLSTATE_UNKNOWN,
|
|
ER(CR_AUTH_PLUGIN_CANNOT_LOAD), name, "invalid type");
|
|
}
|
|
|
|
if ((p= find_plugin(name, type)))
|
|
return p;
|
|
|
|
/* not found, load it */
|
|
return mysql_load_plugin(mysql, name, type, 0);
|
|
}
|
|
|