0
0
mirror of https://gitlab.nic.cz/labs/bird.git synced 2024-12-23 02:01:55 +00:00
bird/lua/common.c
2018-01-23 15:17:01 +01:00

188 lines
4.2 KiB
C

#include "nest/bird.h"
#include "conf/conf.h"
#include "filter/filter.h"
#include "lua.h"
#include <lua.h>
#include <lualib.h>
#include <lauxlib.h>
static int luaB_err(lua_State *L) {
int n = lua_gettop(L);
if (n != 1)
log(L_WARN "bird.err() accepts exactly 1 argument");
if (n < 1)
return 0;
log(L_ERR "%s", lua_tostring(L, 1));
return 0;
}
static int luaB_warn(lua_State *L) {
int n = lua_gettop(L);
if (n != 1)
log(L_WARN "bird.warn() accepts exactly 1 argument");
if (n < 1)
return 0;
log(L_WARN "%s", lua_tostring(L, 1));
return 0;
}
static int luaB_info(lua_State *L) {
int n = lua_gettop(L);
if (n != 1)
log(L_WARN "bird.info() accepts exactly 1 argument");
if (n < 1)
return 0;
log(L_INFO "%s", lua_tostring(L, 1));
return 0;
}
static int luaB_trace(lua_State *L) {
int n = lua_gettop(L);
if (n != 1)
log(L_WARN "bird.trace() accepts exactly 1 argument");
if (n < 1)
return 0;
log(L_TRACE "%s", lua_tostring(L, 1));
return 0;
}
#define lua_sett(L, idx, val, what) do { \
lua_pushstring(L, idx); \
lua_push##what(L, val); \
lua_settable(L, -3); \
} while (0)
#define lua_settablecfunction(L, idx, val) lua_sett(L, idx, val, cfunction)
#define lua_settableinteger(L, idx, val) lua_sett(L, idx, val, integer)
#define lua_settableip4(L, idx, val) lua_sett(L, idx, val, ip4)
#define lua_settablelightuserdata(L, idx, val) lua_sett(L, idx, val, lightuserdata)
#define lua_setglobalcfunction(L, n, val) do { \
lua_pushcfunction(L, val); \
lua_setglobal(L, n); \
} while (0)
static int luaB_generic_concat(lua_State *L) {
int n = lua_gettop(L);
if (n != 2) {
log(L_WARN "__concat needs exactly 2 arguments");
return 0;
}
const char *a, *b;
size_t la, lb;
a = luaL_tolstring(L, 1, &la);
b = luaL_tolstring(L, 2, &lb);
if (a == NULL) {
a = "";
la = 0;
}
if (b == NULL) {
b = "";
lb = 0;
}
char *c = alloca(la + lb + 1);
memcpy(c, a, la);
memcpy(c + la, b, lb);
c[la + lb] = 0;
lua_pushlstring(L, c, la + lb);
return 1;
}
static int luaB_ip4_tostring(lua_State *L) {
int n = lua_gettop(L);
if (n != 1) {
log(L_WARN "__tostring needs exactly 1 argument");
return 0;
}
lua_pushliteral(L, "addr");
lua_gettable(L, 1);
lua_Integer a = lua_tointeger(L, -1);
char c[IP4_MAX_TEXT_LENGTH];
bsnprintf(c, IP4_MAX_TEXT_LENGTH, "%I4", a);
lua_pushstring(L, c);
return 1;
}
static void lua_puship4(lua_State *L, ip4_addr a) {
lua_newtable(L);
lua_settableinteger(L, "addr", a);
lua_newtable(L);
lua_settablecfunction(L, "__tostring", luaB_ip4_tostring);
lua_settablecfunction(L, "__concat", luaB_generic_concat);
lua_setmetatable(L, -2);
}
static lua_bird_state *luaB_getinternalstate(lua_State *L) {
lua_getglobal(L, "bird");
lua_pushstring(L, "_internal_state");
lua_gettable(L, -2);
if (!lua_isuserdata(L, -1))
luaL_error(L, "fatal: bird internal state not found, type %d", lua_type(L, -1));
lua_bird_state *lbs = lua_touserdata(L, -1);
lua_pop(L, 2); /* Pop the user data and then the table. The string is consumed by gettable(). */
return lbs;
}
static int luaB_global_exception(lua_State *L, int value) {
int n = lua_gettop(L);
if (n > 1)
log(L_WARN "Called exception with too many arguments.");
lua_bird_state *lbs = luaB_getinternalstate(L);
lbs->exception = value;
lua_error(L);
return 0;
}
static inline int luaB_accept(lua_State *L) { return luaB_global_exception(L, F_ACCEPT); }
static inline int luaB_reject(lua_State *L) { return luaB_global_exception(L, F_REJECT); }
lua_bird_state *luaB_init(lua_State *L, struct linpool *lp) {
lua_newtable(L);
lua_settablecfunction(L, "err", luaB_err);
lua_settablecfunction(L, "warn", luaB_warn);
lua_settablecfunction(L, "info", luaB_info);
lua_settablecfunction(L, "trace", luaB_trace);
lua_bird_state *lbs = lp_allocz(lp, sizeof(lua_bird_state));
lua_settablelightuserdata(L, "_internal_state", lbs);
lua_settableip4(L, "router_id", config->router_id);
lua_setglobal(L, "bird");
lua_pushcfunction(L, luaB_accept);
lua_setglobal(L, "accept");
lua_pushcfunction(L, luaB_reject);
lua_setglobal(L, "reject");
return lbs;
}
void luaB_push_route(lua_State *L, struct rte *e) {
}