a9505057c5
Commit41390bfe5f
fixed a bug whereby the processing of a MODRESTART command could result in a crash. The approach taken in this fix was to defer the reloading of all modules so that the call stack does not contain functions located in modules that are being reloaded. It did this by scheduling a one-shot timer event for 1 second in the future, in the absense of any better deferral mechanism at the time. Timers are processed by the event loop, which is core to IRCd and cannot be reloaded. Commit59ea3c6753
introduced a mechanism to defer the execution of a function until all events have been processed by the event loop, in order to fix a REHASH bug that could result in a crash due to closing and reopening listener sockets with a pending socket connection event to process after the REHASH was completed. Rework commit41390bfe5f
to use the new deferral mechanism introduced by commit59ea3c6753
and do the same for module reloads.
353 lines
9.7 KiB
C
353 lines
9.7 KiB
C
/* modules/m_modules.c - module for module loading
|
|
* Copyright (c) 2016 Elizabeth Myers <elizabeth@interlinked.me>
|
|
*
|
|
* Permission to use, copy, modify, and/or distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice is present in all copies.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
* DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
|
|
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
|
|
* IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#include "stdinc.h"
|
|
#include "client.h"
|
|
#include "parse.h"
|
|
#include "msg.h"
|
|
#include "modules.h"
|
|
#include "s_newconf.h"
|
|
#include "s_conf.h"
|
|
#include "s_serv.h"
|
|
#include "hash.h"
|
|
#include "ircd.h"
|
|
#include "match.h"
|
|
#include "numeric.h"
|
|
#include "send.h"
|
|
#include "packet.h"
|
|
#include "logger.h"
|
|
|
|
static const char modules_desc[] = "Provides module management commands";
|
|
|
|
static void mo_modload(struct MsgBuf *, struct Client *, struct Client *, int, const char **);
|
|
static void mo_modlist(struct MsgBuf *, struct Client *, struct Client *, int, const char **);
|
|
static void mo_modreload(struct MsgBuf *, struct Client *, struct Client *, int, const char **);
|
|
static void mo_modunload(struct MsgBuf *, struct Client *, struct Client *, int, const char **);
|
|
static void mo_modrestart(struct MsgBuf *, struct Client *, struct Client *, int, const char **);
|
|
|
|
static void me_modload(struct MsgBuf *, struct Client *, struct Client *, int, const char **);
|
|
static void me_modlist(struct MsgBuf *, struct Client *, struct Client *, int, const char **);
|
|
static void me_modreload(struct MsgBuf *, struct Client *, struct Client *, int, const char **);
|
|
static void me_modunload(struct MsgBuf *, struct Client *, struct Client *, int, const char **);
|
|
static void me_modrestart(struct MsgBuf *, struct Client *, struct Client *, int, const char **);
|
|
|
|
static void do_modload(struct Client *, const char *);
|
|
static void do_modunload(struct Client *, const char *);
|
|
static void do_modreload(struct Client *, const char *);
|
|
static void do_modlist(struct Client *, const char *);
|
|
static void do_modrestart(struct Client *);
|
|
|
|
extern void modules_do_reload(void *); /* end of ircd/modules.c */
|
|
extern void modules_do_restart(void *); /* end of ircd/modules.c */
|
|
|
|
struct Message modload_msgtab = {
|
|
"MODLOAD", 0, 0, 0, 0,
|
|
{mg_unreg, mg_not_oper, mg_ignore, mg_ignore, {me_modload, 2}, {mo_modload, 2}}
|
|
};
|
|
|
|
struct Message modunload_msgtab = {
|
|
"MODUNLOAD", 0, 0, 0, 0,
|
|
{mg_unreg, mg_not_oper, mg_ignore, mg_ignore, {me_modunload, 2}, {mo_modunload, 2}}
|
|
};
|
|
|
|
struct Message modreload_msgtab = {
|
|
"MODRELOAD", 0, 0, 0, 0,
|
|
{mg_unreg, mg_not_oper, mg_ignore, mg_ignore, {me_modreload, 2}, {mo_modreload, 2}}
|
|
};
|
|
|
|
struct Message modlist_msgtab = {
|
|
"MODLIST", 0, 0, 0, 0,
|
|
{mg_unreg, mg_not_oper, mg_ignore, mg_ignore, {me_modlist, 0}, {mo_modlist, 0}}
|
|
};
|
|
|
|
struct Message modrestart_msgtab = {
|
|
"MODRESTART", 0, 0, 0, 0,
|
|
{mg_unreg, mg_not_oper, mg_ignore, mg_ignore, {me_modrestart, 0}, {mo_modrestart, 0}}
|
|
};
|
|
|
|
mapi_clist_av1 modules_clist[] = { &modload_msgtab, &modunload_msgtab, &modreload_msgtab, &modlist_msgtab, &modrestart_msgtab, NULL };
|
|
|
|
DECLARE_MODULE_AV2(modules, NULL, NULL, modules_clist, NULL, NULL, NULL, NULL, modules_desc);
|
|
|
|
/* load a module .. */
|
|
static void
|
|
mo_modload(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p, int parc, const char **parv)
|
|
{
|
|
if(!IsOperAdmin(source_p))
|
|
{
|
|
sendto_one(source_p, form_str(ERR_NOPRIVS),
|
|
me.name, source_p->name, "admin");
|
|
return;
|
|
}
|
|
|
|
if(parc > 2)
|
|
{
|
|
sendto_match_servs(source_p, parv[2], CAP_ENCAP, NOCAPS,
|
|
"ENCAP %s MODLOAD %s", parv[2], parv[1]);
|
|
if(match(parv[2], me.name) == 0)
|
|
return;
|
|
}
|
|
|
|
do_modload(source_p, parv[1]);
|
|
}
|
|
|
|
static void
|
|
me_modload(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p, int parc, const char **parv)
|
|
{
|
|
do_modload(source_p, parv[1]);
|
|
}
|
|
|
|
|
|
/* unload a module .. */
|
|
static void
|
|
mo_modunload(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p, int parc, const char **parv)
|
|
{
|
|
if(!IsOperAdmin(source_p))
|
|
{
|
|
sendto_one(source_p, form_str(ERR_NOPRIVS),
|
|
me.name, source_p->name, "admin");
|
|
return;
|
|
}
|
|
|
|
if(parc > 2)
|
|
{
|
|
sendto_match_servs(source_p, parv[2], CAP_ENCAP, NOCAPS,
|
|
"ENCAP %s MODUNLOAD %s", parv[2], parv[1]);
|
|
if(match(parv[2], me.name) == 0)
|
|
return;
|
|
}
|
|
|
|
do_modunload(source_p, parv[1]);
|
|
}
|
|
|
|
static void
|
|
me_modunload(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p, int parc, const char **parv)
|
|
{
|
|
do_modunload(source_p, parv[1]);
|
|
}
|
|
|
|
/* unload and load in one! */
|
|
static void
|
|
mo_modreload(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p, int parc, const char **parv)
|
|
{
|
|
if(!IsOperAdmin(source_p))
|
|
{
|
|
sendto_one(source_p, form_str(ERR_NOPRIVS),
|
|
me.name, source_p->name, "admin");
|
|
return;
|
|
}
|
|
|
|
if(parc > 2)
|
|
{
|
|
sendto_match_servs(source_p, parv[2], CAP_ENCAP, NOCAPS,
|
|
"ENCAP %s MODRELOAD %s", parv[2], parv[1]);
|
|
if(match(parv[2], me.name) == 0)
|
|
return;
|
|
}
|
|
|
|
do_modreload(source_p, parv[1]);
|
|
}
|
|
|
|
static void
|
|
me_modreload(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p, int parc, const char **parv)
|
|
{
|
|
do_modreload(source_p, parv[1]);
|
|
}
|
|
|
|
/* list modules .. */
|
|
static void
|
|
mo_modlist(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p, int parc, const char **parv)
|
|
{
|
|
if(!IsOperAdmin(source_p))
|
|
{
|
|
sendto_one(source_p, form_str(ERR_NOPRIVS),
|
|
me.name, source_p->name, "admin");
|
|
return;
|
|
}
|
|
|
|
if(parc > 2)
|
|
{
|
|
sendto_match_servs(source_p, parv[2], CAP_ENCAP, NOCAPS,
|
|
"ENCAP %s MODLIST %s", parv[2], parv[1]);
|
|
if(match(parv[2], me.name) == 0)
|
|
return;
|
|
}
|
|
|
|
do_modlist(source_p, parc > 1 ? parv[1] : 0);
|
|
}
|
|
|
|
static void
|
|
me_modlist(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p, int parc, const char **parv)
|
|
{
|
|
do_modlist(source_p, parv[1]);
|
|
}
|
|
|
|
/* unload and reload all modules */
|
|
static void
|
|
mo_modrestart(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p, int parc, const char **parv)
|
|
{
|
|
if(!IsOperAdmin(source_p))
|
|
{
|
|
sendto_one(source_p, form_str(ERR_NOPRIVS),
|
|
me.name, source_p->name, "admin");
|
|
return;
|
|
}
|
|
|
|
if(parc > 1)
|
|
{
|
|
sendto_match_servs(source_p, parv[1], CAP_ENCAP, NOCAPS,
|
|
"ENCAP %s MODRESTART", parv[1]);
|
|
if(match(parv[1], me.name) == 0)
|
|
return;
|
|
}
|
|
|
|
do_modrestart(source_p);
|
|
}
|
|
|
|
static void
|
|
me_modrestart(struct MsgBuf *msgbuf_p, struct Client *client_p, struct Client *source_p, int parc, const char **parv)
|
|
{
|
|
do_modrestart(source_p);
|
|
}
|
|
|
|
static void
|
|
do_modload(struct Client *source_p, const char *module)
|
|
{
|
|
char *m_bn = rb_basename(module);
|
|
int origin;
|
|
|
|
if(findmodule_byname(m_bn) != NULL)
|
|
{
|
|
sendto_one_notice(source_p, ":Module %s is already loaded", m_bn);
|
|
rb_free(m_bn);
|
|
return;
|
|
}
|
|
|
|
mod_remember_clicaps();
|
|
|
|
origin = strcmp(module, m_bn) == 0 ? MAPI_ORIGIN_CORE : MAPI_ORIGIN_EXTENSION;
|
|
load_one_module(module, origin, false);
|
|
|
|
mod_notify_clicaps();
|
|
|
|
rb_free(m_bn);
|
|
}
|
|
|
|
static void
|
|
do_modunload(struct Client *source_p, const char *module)
|
|
{
|
|
struct module *mod;
|
|
char *m_bn = rb_basename(module);
|
|
|
|
if((mod = findmodule_byname(m_bn)) == NULL)
|
|
{
|
|
sendto_one_notice(source_p, ":Module %s is not loaded", m_bn);
|
|
rb_free(m_bn);
|
|
return;
|
|
}
|
|
|
|
if(mod->core)
|
|
{
|
|
sendto_one_notice(source_p, ":Module %s is a core module and may not be unloaded", m_bn);
|
|
rb_free(m_bn);
|
|
return;
|
|
}
|
|
|
|
mod_remember_clicaps();
|
|
|
|
if(unload_one_module(m_bn, true) == false)
|
|
sendto_one_notice(source_p, ":Module %s is not loaded", m_bn);
|
|
|
|
mod_notify_clicaps();
|
|
|
|
rb_free(m_bn);
|
|
}
|
|
|
|
static void
|
|
do_modreload(struct Client *source_p, const char *module)
|
|
{
|
|
struct modreload *info = rb_malloc(sizeof *info);
|
|
strcpy(info->module, module);
|
|
strcpy(info->id, source_p->id);
|
|
rb_defer(&modules_do_reload, info);
|
|
}
|
|
|
|
static void
|
|
do_modrestart(struct Client *source_p)
|
|
{
|
|
sendto_one_notice(source_p, ":Reloading all modules");
|
|
|
|
/*
|
|
* If a remote MODRESTART is received, m_encap.so will be reloaded,
|
|
* but ms_encap is in the call stack (it indirectly calls this
|
|
* function). Also, this function is itself in a module.
|
|
*
|
|
* This will go horribly wrong if either module is reloaded to a
|
|
* different address.
|
|
*
|
|
* So, defer the restart to the event loop and return now.
|
|
*/
|
|
rb_defer(&modules_do_restart, NULL);
|
|
}
|
|
|
|
static void
|
|
do_modlist(struct Client *source_p, const char *pattern)
|
|
{
|
|
rb_dlink_node *ptr;
|
|
|
|
RB_DLINK_FOREACH(ptr, module_list.head)
|
|
{
|
|
struct module *mod = ptr->data;
|
|
const char *origin;
|
|
switch (mod->origin)
|
|
{
|
|
case MAPI_ORIGIN_EXTENSION:
|
|
origin = "extension";
|
|
break;
|
|
case MAPI_ORIGIN_CORE:
|
|
origin = "builtin";
|
|
break;
|
|
default:
|
|
origin = "unknown";
|
|
break;
|
|
}
|
|
|
|
if(pattern)
|
|
{
|
|
if(match(pattern, mod->name))
|
|
{
|
|
sendto_one(source_p, form_str(RPL_MODLIST),
|
|
me.name, source_p->name,
|
|
mod->name,
|
|
(unsigned long)(uintptr_t)mod->address, origin,
|
|
mod->core ? " (core)" : "", mod->version, mod->description);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
sendto_one(source_p, form_str(RPL_MODLIST),
|
|
me.name, source_p->name, mod->name,
|
|
(unsigned long)(uintptr_t)mod->address, origin,
|
|
mod->core ? " (core)" : "", mod->version, mod->description);
|
|
}
|
|
}
|
|
|
|
sendto_one(source_p, form_str(RPL_ENDOFMODLIST), me.name, source_p->name);
|
|
}
|