From 091cfcb906e4d78e0ba836b68c9fdd68d2e1ce3d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marek=20Vavru=C5=A1a?= <marek.vavrusa@nic.cz> Date: Mon, 15 Jun 2015 10:26:25 +0200 Subject: [PATCH] daemon/bindings: minimal Lua interface to packet and DNS primitives (wip) --- daemon/bindings/kres.c | 167 +++++++++++++++++++++++++++++++++++++++++ daemon/bindings/kres.h | 30 ++++++++ daemon/main.c | 2 + 3 files changed, 199 insertions(+) create mode 100644 daemon/bindings/kres.c create mode 100644 daemon/bindings/kres.h diff --git a/daemon/bindings/kres.c b/daemon/bindings/kres.c new file mode 100644 index 000000000..986d13b23 --- /dev/null +++ b/daemon/bindings/kres.c @@ -0,0 +1,167 @@ +/* Copyright (C) 2015 CZ.NIC, z.s.p.o. <knot-dns@labs.nic.cz> + + 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 3 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, see <http://www.gnu.org/licenses/>. + */ + +#include "daemon/bindings/kres.h" +#include "daemon/bindings.h" + +/* Metatable list */ +#define META_PKT "kres.meta_pkt" + +/* + * Packet interface + */ + +#define WIRE_FLAGS(X) \ + X(AA,aa) X(AD,ad) X(CD,cd) X(RD,rd) X(QR,qr) X(RA,ra) X(TC,tc) +enum { + #define X(flag, _) WIRE_ ## flag, + WIRE_FLAGS(X) + #undef X +}; +static lookup_table_t wire_flag_names[] = { + #define X(flag, _) { WIRE_ ## flag, #flag }, + WIRE_FLAGS(X) + #undef X +}; + +static int pkt_flag(lua_State *L) +{ + knot_pkt_t *pkt = luaL_checkudata(L, 1, META_PKT); + if (lua_gettop(L) > 1 && lua_isnumber(L, 2)) { + int flag_id = lua_tonumber(L, 2); + switch(flag_id) { + #define X(flag, code) case WIRE_ ## flag: knot_wire_set_ ## code (pkt->wire); break; + WIRE_FLAGS(X) + #undef X + } + } + return 0; +} + +static int pkt_opcode(lua_State *L) +{ + knot_pkt_t *pkt = luaL_checkudata(L, 1, META_PKT); + if (lua_gettop(L) > 1 && lua_isnumber(L, 2)) { + knot_wire_set_opcode(pkt->wire, lua_tonumber(L, 2)); + } + lua_pushnumber(L, knot_wire_get_opcode(pkt->wire)); + return 1; +} + +static int pkt_rcode(lua_State *L) +{ + knot_pkt_t *pkt = luaL_checkudata(L, 1, META_PKT); + if (lua_gettop(L) > 1 && lua_isnumber(L, 2)) { + knot_wire_set_rcode(pkt->wire, lua_tonumber(L, 2)); + } + lua_pushnumber(L, knot_wire_get_rcode(pkt->wire)); + return 1; +} + +static int pkt_qtype(lua_State *L) +{ + knot_pkt_t *pkt = luaL_checkudata(L, 1, META_PKT); + lua_pushnumber(L, knot_pkt_qtype(pkt)); + return 1; +} + +static int pkt_qclass(lua_State *L) +{ + knot_pkt_t *pkt = luaL_checkudata(L, 1, META_PKT); + lua_pushnumber(L, knot_pkt_qclass(pkt)); + return 1; +} + +static int pkt_qname(lua_State *L) +{ + knot_pkt_t *pkt = luaL_checkudata(L, 1, META_PKT); + const knot_dname_t *dname = knot_pkt_qname(pkt); + char dname_str[KNOT_DNAME_MAXLEN]; + knot_dname_to_str(dname_str, dname, sizeof(dname_str)); + lua_pushstring(L, dname_str); + return 1; +} + +#warning TODO: record interfaces + +static int pkt_meta_set(lua_State *L) +{ + static const luaL_Reg pkt_wrap[] = { + { "flag", pkt_flag}, + { "rcode", pkt_rcode}, + { "opcode", pkt_opcode}, + { "qtype", pkt_qtype }, + { "qclass", pkt_qclass }, + { "qname", pkt_qname }, + { NULL, NULL } + }; + luaL_newmetatable(L, META_PKT); + luaL_setfuncs(L, pkt_wrap, 0); + lua_pushvalue(L, -1); + lua_setfield(L, -2, "__index"); + lua_pop(L, 1); + return 0; +} + +static int pkt_meta_get(lua_State *L) +{ + luaL_getmetatable(L, META_PKT); + lua_setmetatable(L, -2); + return 1; +} + +/** + * Resolution context interface. + */ + +#warning TODO: context interface, rplan + +#define WRAP_NUMBER(L, name, val) \ + lua_pushnumber((L), (val)); \ + lua_setfield((L), -2, (name)) + +#define WRAP_CONST(L, name, prefix...) \ + WRAP_NUMBER(L, #name, prefix ## name) + +#define WRAP_LUT(L, prefix, table) \ + lua_newtable(L); \ + for (lookup_table_t *elm = (table); elm->name; ++elm) { \ + WRAP_NUMBER((L), elm->name, elm->id); \ + } \ + lua_setfield((L), -2, (prefix)) + +int lib_kres(lua_State *L) +{ + static const luaL_Reg lib[] = { + { "packet", pkt_meta_get }, + { NULL, NULL } + }; + /* Create module and register functions */ + register_lib(L, "kres", lib); + /* Register states */ + WRAP_CONST(L, NOOP, KNOT_STATE_); + WRAP_CONST(L, CONSUME, KNOT_STATE_); + WRAP_CONST(L, PRODUCE, KNOT_STATE_); + WRAP_CONST(L, DONE, KNOT_STATE_); + WRAP_CONST(L, FAIL, KNOT_STATE_); + /* Register RCODE, OPCODE */ + WRAP_LUT(L, "rcode", knot_rcode_names); + WRAP_LUT(L, "opcode", knot_opcode_names); + WRAP_LUT(L, "wire", wire_flag_names); + /* Register metatables */ + pkt_meta_set(L); + return 1; +} \ No newline at end of file diff --git a/daemon/bindings/kres.h b/daemon/bindings/kres.h new file mode 100644 index 000000000..39209e03a --- /dev/null +++ b/daemon/bindings/kres.h @@ -0,0 +1,30 @@ +/* Copyright (C) 2015 CZ.NIC, z.s.p.o. <knot-dns@labs.nic.cz> + + 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 3 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, see <http://www.gnu.org/licenses/>. + */ + +/** + * Lua-friendly bindings to resolver library parts, + * notably packet parsing and interpretation and operation on primitives like domain names. + */ +#pragma once + +#include "daemon/bindings.h" + +/** + * Load libkres library. + * @param L scriptable + * @return number of packages to load + */ +int lib_kres(lua_State *L); \ No newline at end of file diff --git a/daemon/main.c b/daemon/main.c index 83e82429f..42a08c624 100644 --- a/daemon/main.c +++ b/daemon/main.c @@ -27,6 +27,7 @@ #include "daemon/worker.h" #include "daemon/engine.h" #include "daemon/bindings.h" +#include "daemon/bindings/kres.h" static void tty_read(uv_stream_t *stream, ssize_t nread, const uv_buf_t *buf) { @@ -145,6 +146,7 @@ int main(int argc, char **argv) engine_lualib(&engine, "net", lib_net); engine_lualib(&engine, "cache", lib_cache); engine_lualib(&engine, "event", lib_event); + engine_lualib(&engine, "kres", lib_kres); /* Create main worker. */ struct worker_ctx *worker = mm_alloc(&pool, sizeof(*worker)); -- GitLab