878 lines
20 KiB
C
878 lines
20 KiB
C
/* $NetBSD: lua.c,v 1.14 2014/09/24 14:55:48 mbalmer Exp $ */
|
|
|
|
/*
|
|
* Copyright (c) 2014 by Lourival Vieira Neto <lneto@NetBSD.org>.
|
|
* Copyright (c) 2011 - 2014 by Marc Balmer <mbalmer@NetBSD.org>.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. The name of the Author may not be used to endorse or promote products
|
|
* derived from this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``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 REGENTS OR CONTRIBUTORS 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.
|
|
*/
|
|
|
|
/* Lua device driver */
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/fcntl.h>
|
|
#include <sys/conf.h>
|
|
#include <sys/condvar.h>
|
|
#include <sys/device.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/kmem.h>
|
|
#include <sys/lock.h>
|
|
#include <sys/lua.h>
|
|
#include <sys/module.h>
|
|
#include <sys/mutex.h>
|
|
#include <sys/namei.h>
|
|
#include <sys/queue.h>
|
|
#include <sys/sysctl.h>
|
|
#include <sys/vnode.h>
|
|
#include <sys/cpu.h>
|
|
|
|
#include <lauxlib.h>
|
|
|
|
#include "luavar.h"
|
|
|
|
struct lua_softc {
|
|
device_t sc_dev;
|
|
|
|
kmutex_t sc_lock;
|
|
kcondvar_t sc_inuse_cv;
|
|
bool sc_inuse;
|
|
|
|
/* Locking access to state queues */
|
|
kmutex_t sc_state_lock;
|
|
kcondvar_t sc_state_cv;
|
|
bool sc_state;
|
|
|
|
struct sysctllog *sc_log;
|
|
};
|
|
|
|
static device_t sc_self;
|
|
static bool lua_autoload_on = true;
|
|
static bool lua_require_on = true;
|
|
static bool lua_bytecode_on = false;
|
|
static int lua_verbose;
|
|
static int lua_max_instr;
|
|
|
|
static LIST_HEAD(, lua_state) lua_states;
|
|
static LIST_HEAD(, lua_module) lua_modules;
|
|
|
|
static int lua_match(device_t, cfdata_t, void *);
|
|
static void lua_attach(device_t, device_t, void *);
|
|
static int lua_detach(device_t, int);
|
|
static klua_State *klua_find(const char *);
|
|
static const char *lua_reader(lua_State *, void *, size_t *);
|
|
static void lua_maxcount(lua_State *, lua_Debug *);
|
|
|
|
static int lua_require(lua_State *);
|
|
|
|
CFATTACH_DECL_NEW(lua, sizeof(struct lua_softc),
|
|
lua_match, lua_attach, lua_detach, NULL);
|
|
|
|
dev_type_open(luaopen);
|
|
dev_type_close(luaclose);
|
|
dev_type_ioctl(luaioctl);
|
|
|
|
const struct cdevsw lua_cdevsw = {
|
|
.d_open = luaopen,
|
|
.d_close = luaclose,
|
|
.d_read = noread,
|
|
.d_write = nowrite,
|
|
.d_ioctl = luaioctl,
|
|
.d_stop = nostop,
|
|
.d_tty = notty,
|
|
.d_poll = nopoll,
|
|
.d_mmap = nommap,
|
|
.d_kqfilter = nokqfilter,
|
|
.d_discard = nodiscard,
|
|
.d_flag = D_OTHER | D_MPSAFE
|
|
};
|
|
|
|
struct lua_loadstate {
|
|
struct vnode *vp;
|
|
size_t size;
|
|
off_t off;
|
|
};
|
|
|
|
extern struct cfdriver lua_cd;
|
|
|
|
static int
|
|
lua_match(device_t parent, cfdata_t match, void *aux)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static void
|
|
lua_attach(device_t parent, device_t self, void *aux)
|
|
{
|
|
struct lua_softc *sc;
|
|
const struct sysctlnode *node;
|
|
|
|
if (sc_self)
|
|
return;
|
|
|
|
sc = device_private(self);
|
|
sc->sc_dev = self;
|
|
sc_self = self;
|
|
|
|
mutex_init(&sc->sc_lock, MUTEX_DEFAULT, IPL_VM);
|
|
cv_init(&sc->sc_inuse_cv, "luactl");
|
|
|
|
mutex_init(&sc->sc_state_lock, MUTEX_DEFAULT, IPL_VM);
|
|
cv_init(&sc->sc_state_cv, "luastate");
|
|
|
|
pmf_device_register(self, NULL, NULL);
|
|
|
|
/* Sysctl to provide some control over behaviour */
|
|
sysctl_createv(&sc->sc_log, 0, NULL, &node,
|
|
CTLFLAG_OWNDESC,
|
|
CTLTYPE_NODE, "lua",
|
|
SYSCTL_DESCR("Lua options"),
|
|
NULL, 0, NULL, 0,
|
|
CTL_KERN, CTL_CREATE, CTL_EOL);
|
|
|
|
if (node == NULL) {
|
|
printf(": can't create sysctl node\n");
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* XXX Some of the sysctl values must not be changed after the
|
|
* securelevel has been raised.
|
|
*/
|
|
sysctl_createv(&sc->sc_log, 0, &node, NULL,
|
|
CTLFLAG_READWRITE | CTLFLAG_OWNDESC,
|
|
CTLTYPE_BOOL, "require",
|
|
SYSCTL_DESCR("Enable the require command"),
|
|
NULL, 0, &lua_require_on, 0,
|
|
CTL_CREATE, CTL_EOL);
|
|
|
|
sysctl_createv(&sc->sc_log, 0, &node, NULL,
|
|
CTLFLAG_READWRITE | CTLFLAG_OWNDESC,
|
|
CTLTYPE_BOOL, "autoload",
|
|
SYSCTL_DESCR("Enable automatic load of modules"),
|
|
NULL, 0, &lua_autoload_on, 0,
|
|
CTL_CREATE, CTL_EOL);
|
|
|
|
sysctl_createv(&sc->sc_log, 0, &node, NULL,
|
|
CTLFLAG_READWRITE | CTLFLAG_OWNDESC,
|
|
CTLTYPE_BOOL, "bytecode",
|
|
SYSCTL_DESCR("Enable loading of bytecode"),
|
|
NULL, 0, &lua_bytecode_on, 0,
|
|
CTL_CREATE, CTL_EOL);
|
|
|
|
sysctl_createv(&sc->sc_log, 0, &node, NULL,
|
|
CTLFLAG_READWRITE | CTLFLAG_OWNDESC,
|
|
CTLTYPE_INT, "verbose",
|
|
SYSCTL_DESCR("Enable verbose output"),
|
|
NULL, 0, &lua_verbose, 0,
|
|
CTL_CREATE, CTL_EOL);
|
|
|
|
sysctl_createv(&sc->sc_log, 0, &node, NULL,
|
|
CTLFLAG_READWRITE | CTLFLAG_OWNDESC,
|
|
CTLTYPE_INT, "maxcount",
|
|
SYSCTL_DESCR("Limit maximum instruction count"),
|
|
NULL, 0, &lua_max_instr, 0,
|
|
CTL_CREATE, CTL_EOL);
|
|
|
|
aprint_normal_dev(self, "%s\n", LUA_COPYRIGHT);
|
|
}
|
|
|
|
static int
|
|
lua_detach(device_t self, int flags)
|
|
{
|
|
struct lua_softc *sc;
|
|
struct lua_state *s;
|
|
|
|
sc = device_private(self);
|
|
pmf_device_deregister(self);
|
|
|
|
if (sc->sc_log != NULL) {
|
|
sysctl_teardown(&sc->sc_log);
|
|
sc->sc_log = NULL;
|
|
}
|
|
|
|
/* Traverse the list of states and close them */
|
|
while ((s = LIST_FIRST(&lua_states)) != NULL) {
|
|
LIST_REMOVE(s, lua_next);
|
|
klua_close(s->K);
|
|
if (lua_verbose)
|
|
device_printf(self, "state %s destroyed\n",
|
|
s->lua_name);
|
|
kmem_free(s, sizeof(struct lua_state));
|
|
}
|
|
mutex_destroy(&sc->sc_lock);
|
|
cv_destroy(&sc->sc_inuse_cv);
|
|
mutex_destroy(&sc->sc_state_lock);
|
|
cv_destroy(&sc->sc_state_cv);
|
|
sc_self = NULL;
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
luaopen(dev_t dev, int flag, int mode, struct lwp *l)
|
|
{
|
|
struct lua_softc *sc;
|
|
int error = 0;
|
|
|
|
if (minor(dev) > 0)
|
|
return ENXIO;
|
|
|
|
sc = device_lookup_private(&lua_cd, minor(dev));
|
|
if (sc == NULL)
|
|
return ENXIO;
|
|
|
|
mutex_enter(&sc->sc_lock);
|
|
while (sc->sc_inuse == true) {
|
|
error = cv_wait_sig(&sc->sc_inuse_cv, &sc->sc_lock);
|
|
if (error)
|
|
break;
|
|
}
|
|
if (!error)
|
|
sc->sc_inuse = true;
|
|
mutex_exit(&sc->sc_lock);
|
|
|
|
if (error)
|
|
return error;
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
luaclose(dev_t dev, int flag, int mode, struct lwp *l)
|
|
{
|
|
struct lua_softc *sc;
|
|
|
|
if (minor(dev) > 0)
|
|
return ENXIO;
|
|
sc = device_lookup_private(&lua_cd, minor(dev));
|
|
mutex_enter(&sc->sc_lock);
|
|
sc->sc_inuse = false;
|
|
cv_signal(&sc->sc_inuse_cv);
|
|
mutex_exit(&sc->sc_lock);
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
luaioctl(dev_t dev, u_long cmd, void *data, int flag, struct lwp *l)
|
|
{
|
|
struct lua_softc *sc;
|
|
struct lua_info *info;
|
|
struct lua_create *create;
|
|
struct lua_require *require;
|
|
struct lua_load *load;
|
|
struct lua_state *s;
|
|
struct lua_module *m;
|
|
kauth_cred_t cred;
|
|
struct nameidata nd;
|
|
struct pathbuf *pb;
|
|
struct vattr va;
|
|
struct lua_loadstate ls;
|
|
int error, n;
|
|
klua_State *K;
|
|
|
|
sc = device_lookup_private(&lua_cd, minor(dev));
|
|
if (!device_is_active(sc->sc_dev))
|
|
return EBUSY;
|
|
|
|
switch (cmd) {
|
|
case LUAINFO:
|
|
info = data;
|
|
if (info->states == NULL) {
|
|
info->num_states = 0;
|
|
LIST_FOREACH(s, &lua_states, lua_next)
|
|
info->num_states++;
|
|
} else {
|
|
n = 0;
|
|
LIST_FOREACH(s, &lua_states, lua_next) {
|
|
if (n > info->num_states)
|
|
break;
|
|
copyoutstr(s->lua_name, info->states[n].name,
|
|
MAX_LUA_NAME, NULL);
|
|
copyoutstr(s->lua_desc, info->states[n].desc,
|
|
MAX_LUA_DESC, NULL);
|
|
info->states[n].user = s->K->ks_user;
|
|
n++;
|
|
}
|
|
info->num_states = n;
|
|
}
|
|
break;
|
|
case LUACREATE:
|
|
create = data;
|
|
|
|
if (*create->name == '_') {
|
|
if (lua_verbose)
|
|
device_printf(sc->sc_dev, "names of user "
|
|
"created states must not begin with '_'");
|
|
return ENXIO;
|
|
}
|
|
LIST_FOREACH(s, &lua_states, lua_next)
|
|
if (!strcmp(s->lua_name, create->name)) {
|
|
if (lua_verbose)
|
|
device_printf(sc->sc_dev,
|
|
"state %s exists\n", create->name);
|
|
return EBUSY;
|
|
}
|
|
|
|
K = kluaL_newstate(create->name, create->desc, IPL_NONE);
|
|
K->ks_user = true;
|
|
|
|
if (K == NULL)
|
|
return ENOMEM;
|
|
if (lua_verbose)
|
|
device_printf(sc->sc_dev, "state %s created\n",
|
|
create->name);
|
|
break;
|
|
case LUADESTROY:
|
|
create = data;
|
|
|
|
K = klua_find(create->name);
|
|
|
|
if (K != NULL && (K->ks_user == true)) {
|
|
klua_close(K);
|
|
return 0;
|
|
}
|
|
return EBUSY;
|
|
case LUAREQUIRE: /* 'require' a module in a State */
|
|
require = data;
|
|
LIST_FOREACH(s, &lua_states, lua_next)
|
|
if (!strcmp(s->lua_name, require->state))
|
|
LIST_FOREACH(m, &lua_modules, mod_next)
|
|
if (!strcmp(m->mod_name,
|
|
require->module)) {
|
|
if (lua_verbose)
|
|
device_printf(
|
|
sc->sc_dev,
|
|
"requiring module "
|
|
"%s to state %s\n",
|
|
m->mod_name,
|
|
s->lua_name);
|
|
klua_lock(s->K);
|
|
m->open(s->K->L);
|
|
klua_unlock(s->K);
|
|
m->refcount++;
|
|
LIST_INSERT_HEAD(
|
|
&s->lua_modules, m,
|
|
mod_next);
|
|
return 0;
|
|
}
|
|
return ENXIO;
|
|
case LUALOAD:
|
|
load = data;
|
|
if (strrchr(load->path, '/') == NULL)
|
|
return ENXIO;
|
|
|
|
LIST_FOREACH(s, &lua_states, lua_next)
|
|
if (!strcmp(s->lua_name, load->state)) {
|
|
if (lua_verbose)
|
|
device_printf(sc->sc_dev,
|
|
"loading %s into state %s\n",
|
|
load->path, s->lua_name);
|
|
cred = kauth_cred_get();
|
|
pb = pathbuf_create(load->path);
|
|
if (pb == NULL)
|
|
return ENOMEM;
|
|
NDINIT(&nd, LOOKUP, FOLLOW | NOCHROOT, pb);
|
|
pathbuf_destroy(pb);
|
|
error = vn_open(&nd, FREAD, 0);
|
|
if (error) {
|
|
if (lua_verbose)
|
|
device_printf(sc->sc_dev,
|
|
"error vn_open %d\n",
|
|
error);
|
|
return error;
|
|
}
|
|
error = VOP_GETATTR(nd.ni_vp, &va,
|
|
kauth_cred_get());
|
|
if (error) {
|
|
VOP_UNLOCK(nd.ni_vp);
|
|
vn_close(nd.ni_vp, FREAD,
|
|
kauth_cred_get());
|
|
if (lua_verbose)
|
|
device_printf(sc->sc_dev,
|
|
"erro VOP_GETATTR %d\n",
|
|
error);
|
|
return error;
|
|
}
|
|
if (va.va_type != VREG) {
|
|
VOP_UNLOCK(nd.ni_vp);
|
|
vn_close(nd.ni_vp, FREAD,
|
|
kauth_cred_get());
|
|
return EINVAL;
|
|
}
|
|
ls.vp = nd.ni_vp;
|
|
ls.off = 0L;
|
|
ls.size = va.va_size;
|
|
VOP_UNLOCK(nd.ni_vp);
|
|
klua_lock(s->K);
|
|
error = lua_load(s->K->L, lua_reader, &ls,
|
|
strrchr(load->path, '/') + 1, "bt");
|
|
vn_close(nd.ni_vp, FREAD, cred);
|
|
switch (error) {
|
|
case 0: /* no error */
|
|
break;
|
|
case LUA_ERRSYNTAX:
|
|
if (lua_verbose)
|
|
device_printf(sc->sc_dev,
|
|
"syntax error\n");
|
|
klua_unlock(s->K);
|
|
return EINVAL;
|
|
case LUA_ERRMEM:
|
|
if (lua_verbose)
|
|
device_printf(sc->sc_dev,
|
|
"memory error\n");
|
|
klua_unlock(s->K);
|
|
return ENOMEM;
|
|
default:
|
|
if (lua_verbose)
|
|
device_printf(sc->sc_dev,
|
|
"load error %d: %s\n",
|
|
error,
|
|
lua_tostring(s->K->L, -1));
|
|
klua_unlock(s->K);
|
|
return EINVAL;
|
|
}
|
|
if (lua_max_instr > 0)
|
|
lua_sethook(s->K->L, lua_maxcount,
|
|
LUA_MASKCOUNT, lua_max_instr);
|
|
error = lua_pcall(s->K->L, 0, LUA_MULTRET, 0);
|
|
if (error) {
|
|
if (lua_verbose) {
|
|
device_printf(sc->sc_dev,
|
|
"execution error: %s\n",
|
|
lua_tostring(s->K->L, -1));
|
|
}
|
|
klua_unlock(s->K);
|
|
return EINVAL;
|
|
}
|
|
klua_unlock(s->K);
|
|
return 0;
|
|
}
|
|
return ENXIO;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
lua_require(lua_State *L)
|
|
{
|
|
struct lua_state *s;
|
|
struct lua_module *m, *md;
|
|
const char *module;
|
|
char name[MAXPATHLEN];
|
|
|
|
module = lua_tostring(L, -1);
|
|
md = NULL;
|
|
LIST_FOREACH(m, &lua_modules, mod_next)
|
|
if (!strcmp(m->mod_name, module)) {
|
|
md = m;
|
|
break;
|
|
}
|
|
|
|
if (md == NULL && lua_autoload_on && strchr(module, '/') == NULL) {
|
|
snprintf(name, sizeof name, "lua%s", module);
|
|
if (lua_verbose)
|
|
device_printf(sc_self, "autoload %s\n", name);
|
|
module_autoload(name, MODULE_CLASS_MISC);
|
|
LIST_FOREACH(m, &lua_modules, mod_next)
|
|
if (!strcmp(m->mod_name, module)) {
|
|
md = m;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (md != NULL)
|
|
LIST_FOREACH(s, &lua_states, lua_next)
|
|
if (s->K->L == L) {
|
|
if (lua_verbose)
|
|
device_printf(sc_self,
|
|
"require module %s\n",
|
|
md->mod_name);
|
|
md->open(L);
|
|
md->refcount++;
|
|
LIST_INSERT_HEAD(&s->lua_modules, md, mod_next);
|
|
return 1;
|
|
}
|
|
|
|
lua_pushstring(L, "module not found");
|
|
return lua_error(L);
|
|
}
|
|
|
|
typedef struct {
|
|
size_t size;
|
|
} __packed alloc_header_t;
|
|
|
|
static void *
|
|
lua_alloc(void *ud, void *ptr, size_t osize, size_t nsize)
|
|
{
|
|
void *nptr = NULL;
|
|
|
|
const size_t hdr_size = sizeof(alloc_header_t);
|
|
alloc_header_t *hdr = (alloc_header_t *) ((char *) ptr - hdr_size);
|
|
|
|
if (nsize == 0) { /* freeing */
|
|
if (ptr != NULL)
|
|
kmem_intr_free(hdr, hdr->size);
|
|
} else if (ptr != NULL && nsize <= hdr->size - hdr_size) /* shrinking */
|
|
return ptr; /* don't need to reallocate */
|
|
else { /* creating or expanding */
|
|
km_flag_t sleep = cpu_intr_p() || cpu_softintr_p() ?
|
|
KM_NOSLEEP : KM_SLEEP;
|
|
|
|
size_t alloc_size = nsize + hdr_size;
|
|
alloc_header_t *nhdr = kmem_intr_alloc(alloc_size, sleep);
|
|
if (nhdr == NULL) /* failed to allocate */
|
|
return NULL;
|
|
|
|
nhdr->size = alloc_size;
|
|
nptr = (void *) ((char *) nhdr + hdr_size);
|
|
|
|
if (ptr != NULL) { /* expanding */
|
|
memcpy(nptr, ptr, osize);
|
|
kmem_intr_free(hdr, hdr->size);
|
|
}
|
|
}
|
|
return nptr;
|
|
}
|
|
|
|
static const char *
|
|
lua_reader(lua_State *L, void *data, size_t *size)
|
|
{
|
|
struct lua_loadstate *ls;
|
|
static char buf[1024];
|
|
size_t rsiz;
|
|
|
|
ls = data;
|
|
if (ls->size < sizeof(buf))
|
|
rsiz = ls->size;
|
|
else
|
|
rsiz = sizeof(buf);
|
|
vn_rdwr(UIO_READ, ls->vp, buf, rsiz, ls->off, UIO_SYSSPACE,
|
|
0, curlwp->l_cred, NULL, curlwp);
|
|
if (ls->off == 0L && lua_bytecode_on == false && buf[0] == 0x1b) {
|
|
*size = 0L;
|
|
lua_pushstring(L, "loading of bytecode is not allowed");
|
|
lua_error(L);
|
|
return NULL;
|
|
} else {
|
|
*size = rsiz;
|
|
ls->off += *size;
|
|
ls->size -= *size;
|
|
}
|
|
return buf;
|
|
}
|
|
|
|
static void
|
|
lua_maxcount(lua_State *L, lua_Debug *d)
|
|
{
|
|
lua_pushstring(L, "maximum instruction count exceeded");
|
|
lua_error(L);
|
|
}
|
|
|
|
int
|
|
klua_mod_register(const char *name, lua_CFunction open)
|
|
{
|
|
struct lua_module *m;
|
|
|
|
LIST_FOREACH(m, &lua_modules, mod_next)
|
|
if (!strcmp(m->mod_name, name))
|
|
return EBUSY;
|
|
m = kmem_zalloc(sizeof(struct lua_module), KM_SLEEP);
|
|
strlcpy(m->mod_name, name, LUA_MAX_MODNAME);
|
|
m->open = open;
|
|
m->refcount = 0;
|
|
LIST_INSERT_HEAD(&lua_modules, m, mod_next);
|
|
if (lua_verbose)
|
|
device_printf(sc_self, "registered lua module %s\n", name);
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
klua_mod_unregister(const char *name)
|
|
{
|
|
struct lua_module *m;
|
|
|
|
LIST_FOREACH(m, &lua_modules, mod_next)
|
|
if (!strcmp(m->mod_name, name)) {
|
|
if (m->refcount == 0) {
|
|
LIST_REMOVE(m, mod_next);
|
|
kmem_free(m, sizeof(struct lua_module));
|
|
if (lua_verbose)
|
|
device_printf(sc_self,
|
|
"unregistered lua module %s\n",
|
|
name);
|
|
return 0;
|
|
} else
|
|
return EBUSY;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
klua_State *
|
|
klua_newstate(lua_Alloc f, void *ud, const char *name, const char *desc,
|
|
int ipl)
|
|
{
|
|
klua_State *K;
|
|
struct lua_state *s;
|
|
struct lua_softc *sc;
|
|
int error = 0;
|
|
|
|
s = kmem_zalloc(sizeof(struct lua_state), KM_SLEEP);
|
|
sc = device_private(sc_self);
|
|
mutex_enter(&sc->sc_state_lock);
|
|
while (sc->sc_state == true) {
|
|
error = cv_wait_sig(&sc->sc_state_cv, &sc->sc_state_lock);
|
|
if (error)
|
|
break;
|
|
}
|
|
if (!error)
|
|
sc->sc_state = true;
|
|
mutex_exit(&sc->sc_state_lock);
|
|
|
|
if (error)
|
|
return NULL;
|
|
|
|
K = kmem_zalloc(sizeof(klua_State), KM_SLEEP);
|
|
K->L = lua_newstate(f, ud);
|
|
K->ks_user = false;
|
|
if (K->L == NULL) {
|
|
kmem_free(K, sizeof(klua_State));
|
|
K = NULL;
|
|
goto finish;
|
|
}
|
|
|
|
strlcpy(s->lua_name, name, MAX_LUA_NAME);
|
|
strlcpy(s->lua_desc, desc, MAX_LUA_DESC);
|
|
s->K = K;
|
|
|
|
if (lua_require_on || lua_autoload_on) {
|
|
lua_pushcfunction(K->L, lua_require);
|
|
lua_setglobal(K->L, "require");
|
|
}
|
|
LIST_INSERT_HEAD(&lua_states, s, lua_next);
|
|
|
|
mutex_init(&K->ks_lock, MUTEX_DEFAULT, ipl);
|
|
|
|
finish:
|
|
mutex_enter(&sc->sc_state_lock);
|
|
sc->sc_state = false;
|
|
cv_signal(&sc->sc_state_cv);
|
|
mutex_exit(&sc->sc_state_lock);
|
|
return K;
|
|
}
|
|
|
|
inline klua_State *
|
|
kluaL_newstate(const char *name, const char *desc, int ipl)
|
|
{
|
|
return klua_newstate(lua_alloc, NULL, name, desc, ipl);
|
|
}
|
|
|
|
void
|
|
klua_close(klua_State *K)
|
|
{
|
|
struct lua_state *s;
|
|
struct lua_softc *sc;
|
|
struct lua_module *m;
|
|
int error = 0;
|
|
|
|
/* XXX consider registering a handler instead of a fixed name. */
|
|
lua_getglobal(K->L, "onClose");
|
|
if (lua_isfunction(K->L, -1))
|
|
lua_pcall(K->L, -1, 0, 0);
|
|
|
|
sc = device_private(sc_self);
|
|
mutex_enter(&sc->sc_state_lock);
|
|
while (sc->sc_state == true) {
|
|
error = cv_wait_sig(&sc->sc_state_cv, &sc->sc_state_lock);
|
|
if (error)
|
|
break;
|
|
}
|
|
if (!error)
|
|
sc->sc_state = true;
|
|
mutex_exit(&sc->sc_state_lock);
|
|
|
|
if (error)
|
|
return; /* Nothing we can do... */
|
|
|
|
LIST_FOREACH(s, &lua_states, lua_next)
|
|
if (s->K == K) {
|
|
LIST_REMOVE(s, lua_next);
|
|
LIST_FOREACH(m, &s->lua_modules, mod_next)
|
|
m->refcount--;
|
|
kmem_free(s, sizeof(struct lua_state));
|
|
}
|
|
|
|
lua_close(K->L);
|
|
mutex_destroy(&K->ks_lock);
|
|
kmem_free(K, sizeof(klua_State));
|
|
|
|
mutex_enter(&sc->sc_state_lock);
|
|
sc->sc_state = false;
|
|
cv_signal(&sc->sc_state_cv);
|
|
mutex_exit(&sc->sc_state_lock);
|
|
}
|
|
|
|
static klua_State *
|
|
klua_find(const char *name)
|
|
{
|
|
struct lua_state *s;
|
|
struct lua_softc *sc;
|
|
klua_State *K;
|
|
int error = 0;
|
|
|
|
K = NULL;
|
|
sc = device_private(sc_self);
|
|
mutex_enter(&sc->sc_state_lock);
|
|
while (sc->sc_state == true) {
|
|
error = cv_wait_sig(&sc->sc_state_cv, &sc->sc_state_lock);
|
|
if (error)
|
|
break;
|
|
}
|
|
if (!error)
|
|
sc->sc_state = true;
|
|
mutex_exit(&sc->sc_state_lock);
|
|
|
|
if (error)
|
|
return NULL;
|
|
|
|
LIST_FOREACH(s, &lua_states, lua_next)
|
|
if (!strcmp(s->lua_name, name)) {
|
|
K = s->K;
|
|
break;
|
|
}
|
|
|
|
mutex_enter(&sc->sc_state_lock);
|
|
sc->sc_state = false;
|
|
cv_signal(&sc->sc_state_cv);
|
|
mutex_exit(&sc->sc_state_lock);
|
|
return K;
|
|
}
|
|
|
|
inline void
|
|
klua_lock(klua_State *K)
|
|
{
|
|
mutex_enter(&K->ks_lock);
|
|
}
|
|
|
|
inline void
|
|
klua_unlock(klua_State *K)
|
|
{
|
|
mutex_exit(&K->ks_lock);
|
|
}
|
|
|
|
MODULE(MODULE_CLASS_MISC, lua, NULL);
|
|
|
|
#ifdef _MODULE
|
|
static const struct cfiattrdata luabus_iattrdata = {
|
|
"luabus", 0, { { NULL, NULL, 0 },}
|
|
};
|
|
|
|
static const struct cfiattrdata *const lua_attrs[] = {
|
|
&luabus_iattrdata, NULL
|
|
};
|
|
|
|
CFDRIVER_DECL(lua, DV_DULL, lua_attrs);
|
|
extern struct cfattach lua_ca;
|
|
static int lualoc[] = {
|
|
-1,
|
|
-1,
|
|
-1
|
|
};
|
|
|
|
static struct cfdata lua_cfdata[] = {
|
|
{
|
|
.cf_name = "lua",
|
|
.cf_atname = "lua",
|
|
.cf_unit = 0,
|
|
.cf_fstate = FSTATE_STAR,
|
|
.cf_loc = lualoc,
|
|
.cf_flags = 0,
|
|
.cf_pspec = NULL,
|
|
},
|
|
{ NULL, NULL, 0, FSTATE_NOTFOUND, NULL, 0, NULL }
|
|
};
|
|
#endif
|
|
|
|
static int
|
|
lua_modcmd(modcmd_t cmd, void *opaque)
|
|
{
|
|
#ifdef _MODULE
|
|
devmajor_t cmajor, bmajor;
|
|
int error = 0;
|
|
|
|
cmajor = bmajor = NODEVMAJOR;
|
|
#endif
|
|
switch (cmd) {
|
|
case MODULE_CMD_INIT:
|
|
#ifdef _MODULE
|
|
error = config_cfdriver_attach(&lua_cd);
|
|
if (error)
|
|
return error;
|
|
|
|
error = config_cfattach_attach(lua_cd.cd_name,
|
|
&lua_ca);
|
|
if (error) {
|
|
config_cfdriver_detach(&lua_cd);
|
|
aprint_error("%s: unable to register cfattach\n",
|
|
lua_cd.cd_name);
|
|
return error;
|
|
}
|
|
error = config_cfdata_attach(lua_cfdata, 1);
|
|
if (error) {
|
|
config_cfattach_detach(lua_cd.cd_name,
|
|
&lua_ca);
|
|
config_cfdriver_detach(&lua_cd);
|
|
aprint_error("%s: unable to register cfdata\n",
|
|
lua_cd.cd_name);
|
|
return error;
|
|
}
|
|
error = devsw_attach(lua_cd.cd_name, NULL, &bmajor,
|
|
&lua_cdevsw, &cmajor);
|
|
if (error) {
|
|
aprint_error("%s: unable to register devsw\n",
|
|
lua_cd.cd_name);
|
|
config_cfattach_detach(lua_cd.cd_name, &lua_ca);
|
|
config_cfdriver_detach(&lua_cd);
|
|
return error;
|
|
}
|
|
config_attach_pseudo(lua_cfdata);
|
|
#endif
|
|
return 0;
|
|
case MODULE_CMD_FINI:
|
|
#ifdef _MODULE
|
|
error = config_cfdata_detach(lua_cfdata);
|
|
if (error)
|
|
return error;
|
|
|
|
config_cfattach_detach(lua_cd.cd_name, &lua_ca);
|
|
config_cfdriver_detach(&lua_cd);
|
|
devsw_detach(NULL, &lua_cdevsw);
|
|
#endif
|
|
return 0;
|
|
case MODULE_CMD_AUTOUNLOAD:
|
|
/* no auto-unload */
|
|
return EBUSY;
|
|
default:
|
|
return ENOTTY;
|
|
}
|
|
}
|