2008-05-20 15:39:47 +02:00
|
|
|
/*
|
|
|
|
* lua.h - Lua configuration management header
|
|
|
|
*
|
|
|
|
* Copyright © 2008 Julien Danjou <julien@danjou.info>
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program 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 General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along
|
|
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef AWESOME_LUA_H
|
|
|
|
#define AWESOME_LUA_H
|
|
|
|
|
2008-06-17 00:30:53 +02:00
|
|
|
#include <ev.h>
|
2008-05-20 19:02:40 +02:00
|
|
|
#include <lua.h>
|
|
|
|
#include <lauxlib.h>
|
2008-05-20 15:39:47 +02:00
|
|
|
|
|
|
|
#include "common/util.h"
|
2008-07-02 11:04:17 +02:00
|
|
|
#include "common/draw.h"
|
2008-05-20 15:39:47 +02:00
|
|
|
|
2008-06-04 19:21:21 +02:00
|
|
|
/** Object types */
|
|
|
|
typedef enum
|
|
|
|
{
|
|
|
|
AWESOME_TYPE_STATUSBAR = 1,
|
|
|
|
AWESOME_TYPE_TITLEBAR
|
|
|
|
} awesome_type_t;
|
|
|
|
|
2008-05-20 15:39:47 +02:00
|
|
|
/** Type for Lua function */
|
|
|
|
typedef int luaA_function;
|
|
|
|
|
2008-06-18 18:31:35 +02:00
|
|
|
#define DO_LUA_NEW(decl, type, prefix, lua_type, type_ref) \
|
|
|
|
decl int \
|
|
|
|
luaA_##prefix##_userdata_new(lua_State *L, type *p) \
|
|
|
|
{ \
|
|
|
|
type **pp = lua_newuserdata(L, sizeof(type *)); \
|
|
|
|
*pp = p; \
|
|
|
|
type_ref(pp); \
|
|
|
|
return luaA_settype(L, lua_type); \
|
|
|
|
}
|
|
|
|
|
2008-06-16 11:31:38 +02:00
|
|
|
#define DO_LUA_GC(type, prefix, lua_type, type_unref) \
|
|
|
|
static int \
|
|
|
|
luaA_##prefix##_gc(lua_State *L) \
|
|
|
|
{ \
|
|
|
|
type **p = luaA_checkudata(L, 1, lua_type); \
|
|
|
|
type_unref(p); \
|
|
|
|
*p = NULL; \
|
|
|
|
return 0; \
|
|
|
|
}
|
|
|
|
|
2008-06-16 11:37:55 +02:00
|
|
|
#define DO_LUA_EQ(type, prefix, lua_type) \
|
|
|
|
static int \
|
|
|
|
luaA_##prefix##_eq(lua_State *L) \
|
|
|
|
{ \
|
|
|
|
type **p1 = luaA_checkudata(L, 1, lua_type); \
|
|
|
|
type **p2 = luaA_checkudata(L, 2, lua_type); \
|
|
|
|
lua_pushboolean(L, (*p1 == *p2)); \
|
|
|
|
return 1; \
|
|
|
|
}
|
|
|
|
|
2008-05-20 15:39:47 +02:00
|
|
|
#define luaA_dostring(L, cmd) \
|
|
|
|
do { \
|
2008-07-30 18:20:19 +02:00
|
|
|
if(a_strlen(cmd)) \
|
2008-05-20 15:39:47 +02:00
|
|
|
if(luaL_dostring(L, cmd)) \
|
2008-05-23 22:53:59 +02:00
|
|
|
warn("error executing Lua code: %s", \
|
2008-05-20 15:39:47 +02:00
|
|
|
lua_tostring(L, -1)); \
|
|
|
|
} while(0)
|
|
|
|
|
|
|
|
#define luaA_checktable(L, n) \
|
|
|
|
do { \
|
|
|
|
if(!lua_istable(L, n)) \
|
|
|
|
luaL_typerror(L, n, "table"); \
|
|
|
|
} while(0)
|
|
|
|
|
|
|
|
#define luaA_checkfunction(L, n) \
|
|
|
|
do { \
|
|
|
|
if(!lua_isfunction(L, n)) \
|
|
|
|
luaL_typerror(L, n, "function"); \
|
|
|
|
} while(0)
|
|
|
|
|
|
|
|
#define luaA_checkscreen(screen) \
|
|
|
|
do { \
|
|
|
|
if(screen < 0 || screen >= globalconf.screens_info->nscreen) \
|
2008-08-08 14:37:27 +02:00
|
|
|
luaL_error(L, "invalid screen number: %d", screen); \
|
2008-05-20 15:39:47 +02:00
|
|
|
} while(0)
|
|
|
|
|
2008-06-04 18:27:10 +02:00
|
|
|
/** Check that an object is not a NULL reference.
|
|
|
|
* \param L The Lua state.
|
|
|
|
* \param ud The index of the object in the stack.
|
|
|
|
* \param tname The type name.
|
|
|
|
* \return A pointer to the object.
|
|
|
|
*/
|
|
|
|
static inline void *
|
|
|
|
luaA_checkudata(lua_State *L, int ud, const char *tname)
|
|
|
|
{
|
|
|
|
void **p = luaL_checkudata(L, ud, tname);
|
|
|
|
if(*p)
|
|
|
|
return p;
|
|
|
|
luaL_error(L, "invalid object");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2008-06-12 13:27:45 +02:00
|
|
|
static inline bool
|
|
|
|
luaA_checkboolean(lua_State *L, int n)
|
|
|
|
{
|
|
|
|
if(!lua_isboolean(L, n))
|
|
|
|
luaL_typerror(L, n, "boolean");
|
|
|
|
return lua_toboolean(L, n);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool
|
|
|
|
luaA_optboolean(lua_State *L, int idx, bool def)
|
|
|
|
{
|
|
|
|
return luaL_opt(L, luaA_checkboolean, idx, def);
|
|
|
|
}
|
|
|
|
|
2008-05-20 15:39:47 +02:00
|
|
|
static inline lua_Number
|
|
|
|
luaA_getopt_number(lua_State *L, int idx, const char *name, lua_Number def)
|
|
|
|
{
|
|
|
|
lua_getfield(L, idx, name);
|
|
|
|
return luaL_optnumber(L, -1, def);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline const char *
|
2008-06-25 17:33:05 +02:00
|
|
|
luaA_getopt_lstring(lua_State *L, int idx, const char *name, const char *def, size_t *len)
|
2008-05-20 15:39:47 +02:00
|
|
|
{
|
|
|
|
lua_getfield(L, idx, name);
|
2008-06-23 14:01:33 +02:00
|
|
|
return luaL_optlstring(L, -1, def, len);
|
2008-05-20 15:39:47 +02:00
|
|
|
}
|
|
|
|
|
2008-06-25 17:33:05 +02:00
|
|
|
static inline const char *
|
|
|
|
luaA_getopt_string(lua_State *L, int idx, const char *name, const char *def)
|
|
|
|
{
|
|
|
|
return luaA_getopt_lstring(L, idx, name, def, NULL);
|
|
|
|
}
|
|
|
|
|
2008-06-12 13:27:45 +02:00
|
|
|
static inline bool
|
|
|
|
luaA_getopt_boolean(lua_State *L, int idx, const char *name, bool def)
|
|
|
|
{
|
|
|
|
lua_getfield(L, idx, name);
|
|
|
|
return luaA_optboolean(L, -1, def);
|
|
|
|
}
|
|
|
|
|
2008-05-20 15:39:47 +02:00
|
|
|
static inline int
|
|
|
|
luaA_settype(lua_State *L, const char *type)
|
|
|
|
{
|
|
|
|
luaL_getmetatable(L, type);
|
|
|
|
lua_setmetatable(L, -2);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2008-07-01 19:25:58 +02:00
|
|
|
static inline int
|
|
|
|
luaA_usemetatable(lua_State *L, int idxobj, int idxfield)
|
|
|
|
{
|
|
|
|
lua_getmetatable(L, idxobj);
|
|
|
|
lua_pushvalue(L, idxfield);
|
|
|
|
lua_rawget(L, -2);
|
|
|
|
if (!lua_isnil(L, -1))
|
|
|
|
{
|
|
|
|
lua_remove(L, -2);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
lua_pop(L, 2);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-07-29 10:43:11 +02:00
|
|
|
/** Register a function.
|
|
|
|
* \param L The Lua stack.
|
|
|
|
* \param fct A luaA_function address: it will be filled with the luaA_function
|
|
|
|
* registered. If the adresse point to an already registered function, it will
|
|
|
|
* be unregistered.
|
|
|
|
* \return Always 0.
|
|
|
|
*/
|
2008-07-09 12:12:52 +02:00
|
|
|
static inline int
|
|
|
|
luaA_registerfct(lua_State *L, luaA_function *fct)
|
|
|
|
{
|
|
|
|
luaA_checkfunction(L, -1);
|
|
|
|
if(*fct != LUA_REFNIL)
|
|
|
|
luaL_unref(L, LUA_REGISTRYINDEX, *fct);
|
|
|
|
*fct = luaL_ref(L, LUA_REGISTRYINDEX);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-07-29 10:43:11 +02:00
|
|
|
/** Execute an Lua function.
|
|
|
|
* \param L The Lua stack.
|
|
|
|
* \param f The Lua function to execute.
|
|
|
|
* \param nargs The number of arguments for the Lua function.
|
|
|
|
* \param nret The number of returned value from the Lua function.
|
|
|
|
* \return True on no error, false otherwise.
|
|
|
|
*/
|
|
|
|
static inline bool
|
|
|
|
luaA_dofunction(lua_State *L, luaA_function f, int nargs, int nret)
|
|
|
|
{
|
|
|
|
if(f != LUA_REFNIL)
|
|
|
|
{
|
|
|
|
lua_rawgeti(L, LUA_REGISTRYINDEX, f);
|
|
|
|
if(nargs)
|
|
|
|
lua_insert(L, - (nargs + 1));
|
|
|
|
if(lua_pcall(L, nargs, nret, 0))
|
|
|
|
{
|
|
|
|
warn("error running function: %s",
|
|
|
|
lua_tostring(L, -1));
|
|
|
|
lua_pop(L, 1);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-08-07 15:48:26 +02:00
|
|
|
int luaA_otable_index(lua_State *);
|
|
|
|
|
|
|
|
/** Create a new object table with a metatable.
|
|
|
|
* This is useful to compare table with objects (udata) as keys.
|
|
|
|
* \param L The Lua stack.
|
|
|
|
* \return The number of elements pushed on stack.
|
|
|
|
*/
|
|
|
|
static inline int
|
|
|
|
luaA_otable_new(lua_State *L)
|
|
|
|
{
|
|
|
|
/* Our object */
|
|
|
|
lua_newtable(L);
|
|
|
|
/* The meta table */
|
|
|
|
lua_newtable(L);
|
|
|
|
lua_pushcfunction(L, luaA_otable_index);
|
|
|
|
/* Register index into the metatable */
|
|
|
|
lua_setfield(L, -2, "__index");
|
|
|
|
/* Set the meta table */
|
|
|
|
lua_setmetatable(L, -2);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2008-06-08 21:49:15 +02:00
|
|
|
void luaA_init(void);
|
2008-07-08 14:07:56 +02:00
|
|
|
void luaA_parserc(const char *);
|
2008-06-18 18:31:35 +02:00
|
|
|
void luaA_pushpointer(lua_State *, void *, awesome_type_t);
|
2008-06-16 23:47:14 +02:00
|
|
|
void luaA_cs_init(void);
|
|
|
|
void luaA_cs_cleanup(void);
|
2008-06-17 00:30:53 +02:00
|
|
|
void luaA_on_timer(EV_P_ ev_timer *w, int revents);
|
2008-08-10 16:14:16 +02:00
|
|
|
void luaA_pushcolor(lua_State *, const xcolor_t *);
|
|
|
|
|
|
|
|
static inline int
|
|
|
|
luaA_generic_pairs(lua_State *L)
|
|
|
|
{
|
|
|
|
lua_pushvalue(L, lua_upvalueindex(1)); /* return generator, */
|
|
|
|
lua_pushvalue(L, 1); /* state, */
|
|
|
|
lua_pushnil(L); /* and initial value */
|
|
|
|
return 3;
|
|
|
|
}
|
2008-06-04 19:21:21 +02:00
|
|
|
|
Alternative export method of arrays from C to Lua
The patch is mainly to export client_array_t object to Lua,
but can be used to export any ..._array_t object.
The idea: export to Lua not a table, but userdata with
metamethods to get/set/define length of ..._array_t object
directly.
Now when I get clients field from tag object C code
creates full copy of client_array_t structure into Lua table.
It takes traversing a whole array of data.
I did it in other way: userdata is exported, with __index,
__newindex, and __len meta-methods defined, and Lua
script gains direct access to client_array_t C-array:
it can get client object, get length of array and assign
client objects to some index in C-array.
Pros:
No overhead of creation a copy of C-structure into Lua-table:
if I want just to test a number of clients for a tag, I don't need
a whole loop to build table, I just want to read clients->len field,
and I do so via __len meta-method.
Also if I want to get some client from tags.clients, I don't need
to create ALL clients Lua-objects, I just get client_t C-struct
and create Lua-object from it. Just in place.
So Lua-loop enuming all tag.clients is not 2 loops internally
(first create copy of tag.clients into Lua-table, then enum this table),
but only one, and if I break out of loop in the middle, I create
only some client Lua-objects, not all of them from tag.clients.
Contras:
As far as clients field is not a table, I cant use pairs/ipairs
and other table functions for it.
But it can be implemented in other way:
for k,c pairs(tag.clients) => for k = 1, #tag.clients,
table.insert(tag.clients, client) => tag.clients[#tag.clients+1] = client
etc.
One more Pro now:
As far as tag.clients in current implementation returns copy of data
table.insert doesn't do what's expected: it doesn't really add client
into tag.clients "array".
With my implementation client is added as expected, as we work with
client_array_t structure directly.
Signed-off-by: Julien Danjou <julien@danjou.info>
2008-08-10 14:05:50 +02:00
|
|
|
#define DO_LUA_EXPORT_ARRAY(pfx, typename, type, atype, ctor) \
|
|
|
|
static inline int \
|
|
|
|
luaA_##pfx##_array_index(lua_State *L) \
|
|
|
|
{ \
|
|
|
|
atype *value = lua_touserdata(L, 1); \
|
|
|
|
type **obj; \
|
|
|
|
int idx; \
|
|
|
|
\
|
|
|
|
if ((obj = lua_touserdata(L, 2))) \
|
|
|
|
{ \
|
|
|
|
for (idx = 0; idx < value->len; idx++) \
|
|
|
|
if (value->tab[idx] == *obj) \
|
|
|
|
return ctor(L, value->tab[idx]); \
|
|
|
|
return 0; \
|
|
|
|
} \
|
|
|
|
\
|
|
|
|
idx = luaL_checknumber(L, 2) - 1; \
|
|
|
|
if (idx < 0 || idx >= value->len) \
|
|
|
|
return 0; \
|
|
|
|
\
|
|
|
|
return ctor(L, value->tab[idx]); \
|
|
|
|
} \
|
|
|
|
static inline int \
|
|
|
|
luaA_##pfx##_array_newindex(lua_State *L) \
|
|
|
|
{ \
|
|
|
|
atype *value = lua_touserdata(L, 1); \
|
|
|
|
int idx = luaL_checknumber(L, 2); \
|
|
|
|
type **elem = luaL_checkudata(L, 3, #pfx); \
|
|
|
|
pfx##_array_splice(value, idx - 1, 1, elem, 1); \
|
|
|
|
return 0; \
|
|
|
|
} \
|
|
|
|
static inline int \
|
|
|
|
luaA_##pfx##_array_len(lua_State *L) \
|
|
|
|
{ \
|
|
|
|
atype *value = lua_touserdata(L, 1); \
|
|
|
|
lua_pushnumber(L, value->len); \
|
|
|
|
return 1; \
|
|
|
|
} \
|
|
|
|
static inline int \
|
|
|
|
luaA_##pfx##_array_tostring(lua_State *L) \
|
|
|
|
{ \
|
|
|
|
atype *value = lua_touserdata(L, 1); \
|
|
|
|
lua_pushfstring(L, "["typename" udata(%p)]", value); \
|
|
|
|
return 1; \
|
|
|
|
} \
|
2008-08-10 16:14:35 +02:00
|
|
|
static inline int \
|
|
|
|
luaA_##pfx##_array_next(lua_State *L) \
|
|
|
|
{ \
|
|
|
|
atype *value = lua_touserdata(L, 1); \
|
|
|
|
if(value) \
|
|
|
|
{ \
|
|
|
|
lua_settop(L, 2); \
|
|
|
|
if(lua_isnumber(L, 2)) \
|
|
|
|
{ \
|
|
|
|
int idx = lua_tonumber(L, 2); \
|
|
|
|
if(idx >= 0 && idx < value->len) \
|
|
|
|
return ctor(L, value->tab[idx]); \
|
|
|
|
} \
|
|
|
|
else if(lua_isnil(L, 2)) \
|
|
|
|
{ \
|
|
|
|
if(value->len) \
|
|
|
|
return ctor(L, value->tab[0]); \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
return 0; \
|
|
|
|
} \
|
Alternative export method of arrays from C to Lua
The patch is mainly to export client_array_t object to Lua,
but can be used to export any ..._array_t object.
The idea: export to Lua not a table, but userdata with
metamethods to get/set/define length of ..._array_t object
directly.
Now when I get clients field from tag object C code
creates full copy of client_array_t structure into Lua table.
It takes traversing a whole array of data.
I did it in other way: userdata is exported, with __index,
__newindex, and __len meta-methods defined, and Lua
script gains direct access to client_array_t C-array:
it can get client object, get length of array and assign
client objects to some index in C-array.
Pros:
No overhead of creation a copy of C-structure into Lua-table:
if I want just to test a number of clients for a tag, I don't need
a whole loop to build table, I just want to read clients->len field,
and I do so via __len meta-method.
Also if I want to get some client from tags.clients, I don't need
to create ALL clients Lua-objects, I just get client_t C-struct
and create Lua-object from it. Just in place.
So Lua-loop enuming all tag.clients is not 2 loops internally
(first create copy of tag.clients into Lua-table, then enum this table),
but only one, and if I break out of loop in the middle, I create
only some client Lua-objects, not all of them from tag.clients.
Contras:
As far as clients field is not a table, I cant use pairs/ipairs
and other table functions for it.
But it can be implemented in other way:
for k,c pairs(tag.clients) => for k = 1, #tag.clients,
table.insert(tag.clients, client) => tag.clients[#tag.clients+1] = client
etc.
One more Pro now:
As far as tag.clients in current implementation returns copy of data
table.insert doesn't do what's expected: it doesn't really add client
into tag.clients "array".
With my implementation client is added as expected, as we work with
client_array_t structure directly.
Signed-off-by: Julien Danjou <julien@danjou.info>
2008-08-10 14:05:50 +02:00
|
|
|
static inline void \
|
|
|
|
luaA_##pfx##_array_export(lua_State *L, atype *arr) \
|
|
|
|
{ \
|
|
|
|
lua_pushlightuserdata(L, arr); \
|
|
|
|
lua_newtable(L); \
|
|
|
|
lua_pushcfunction(L, luaA_##pfx##_array_index); \
|
|
|
|
lua_setfield(L, -2, "__index"); \
|
|
|
|
lua_pushcfunction(L, luaA_##pfx##_array_tostring); \
|
|
|
|
lua_setfield(L, -2, "__tostring"); \
|
2008-08-10 16:14:35 +02:00
|
|
|
lua_pushcfunction(L, luaA_##pfx##_array_next); \
|
|
|
|
lua_setfield(L, -2, "__next"); \
|
|
|
|
lua_pushcfunction(L, luaA_##pfx##_array_next); \
|
|
|
|
lua_pushcclosure(L, luaA_generic_pairs, 1); \
|
|
|
|
lua_setfield(L, -2, "__pairs"); \
|
Alternative export method of arrays from C to Lua
The patch is mainly to export client_array_t object to Lua,
but can be used to export any ..._array_t object.
The idea: export to Lua not a table, but userdata with
metamethods to get/set/define length of ..._array_t object
directly.
Now when I get clients field from tag object C code
creates full copy of client_array_t structure into Lua table.
It takes traversing a whole array of data.
I did it in other way: userdata is exported, with __index,
__newindex, and __len meta-methods defined, and Lua
script gains direct access to client_array_t C-array:
it can get client object, get length of array and assign
client objects to some index in C-array.
Pros:
No overhead of creation a copy of C-structure into Lua-table:
if I want just to test a number of clients for a tag, I don't need
a whole loop to build table, I just want to read clients->len field,
and I do so via __len meta-method.
Also if I want to get some client from tags.clients, I don't need
to create ALL clients Lua-objects, I just get client_t C-struct
and create Lua-object from it. Just in place.
So Lua-loop enuming all tag.clients is not 2 loops internally
(first create copy of tag.clients into Lua-table, then enum this table),
but only one, and if I break out of loop in the middle, I create
only some client Lua-objects, not all of them from tag.clients.
Contras:
As far as clients field is not a table, I cant use pairs/ipairs
and other table functions for it.
But it can be implemented in other way:
for k,c pairs(tag.clients) => for k = 1, #tag.clients,
table.insert(tag.clients, client) => tag.clients[#tag.clients+1] = client
etc.
One more Pro now:
As far as tag.clients in current implementation returns copy of data
table.insert doesn't do what's expected: it doesn't really add client
into tag.clients "array".
With my implementation client is added as expected, as we work with
client_array_t structure directly.
Signed-off-by: Julien Danjou <julien@danjou.info>
2008-08-10 14:05:50 +02:00
|
|
|
lua_pushcfunction(L, luaA_##pfx##_array_newindex); \
|
|
|
|
lua_setfield(L, -2, "__newindex"); \
|
|
|
|
lua_pushcfunction(L, luaA_##pfx##_array_len); \
|
|
|
|
lua_setfield(L, -2, "__len"); \
|
|
|
|
lua_setmetatable(L, -2); \
|
|
|
|
}
|
|
|
|
|
2008-05-20 15:39:47 +02:00
|
|
|
#endif
|
|
|
|
// vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80
|