You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
lua-zmq/socket.nobj.lua

265 lines
6.6 KiB
Lua

-- Copyright (c) 2010 by Robert G. Jakabosky <bobby@sharedrealm.com>
--
-- Permission is hereby granted, free of charge, to any person obtaining a copy
-- of this software and associated documentation files (the "Software"), to deal
-- in the Software without restriction, including without limitation the rights
-- to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
-- copies of the Software, and to permit persons to whom the Software is
-- furnished to do so, subject to the following conditions:
--
-- The above copyright notice and this permission notice shall be included in
-- all copies or substantial portions of the Software.
--
-- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
-- IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
-- FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
-- AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
-- LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
-- OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
-- THE SOFTWARE.
object "ZMQ_Socket" {
c_source [[
/* detect zmq version >= 2.1.0 */
#define VERSION_2_1 0
#if defined(ZMQ_VERSION)
#if (ZMQ_VERSION >= ZMQ_MAKE_VERSION(2,1,0))
#undef VERSION_2_1
#define VERSION_2_1 1
#endif
#endif
typedef void * ZMQ_Socket;
#if VERSION_2_1
#ifdef _WIN32
#include <winsock2.h>
typedef SOCKET socket_t;
#else
typedef int socket_t;
#endif
#endif
/* socket option types. */
#define OPT_TYPE_NONE 0
#define OPT_TYPE_INT 1
#define OPT_TYPE_UINT32 2
#define OPT_TYPE_UINT64 3
#define OPT_TYPE_INT64 4
#define OPT_TYPE_STR 5
#define OPT_TYPE_FD 6
static const int opt_types[] = {
OPT_TYPE_NONE, /* unused */
OPT_TYPE_UINT64, /* ZMQ_HWM */
OPT_TYPE_INT64, /* ZMQ_SWAP */
OPT_TYPE_UINT64, /* ZMQ_AFFINITY */
OPT_TYPE_STR, /* ZMQ_IDENTITY */
OPT_TYPE_STR, /* ZMQ_SUBSCRIBE */
OPT_TYPE_STR, /* ZMQ_UNSUBSCRIBE */
OPT_TYPE_INT64, /* ZMQ_RATE */
OPT_TYPE_INT64, /* ZMQ_RECOVERY_IVL */
OPT_TYPE_INT64, /* ZMQ_MCAST_LOOP */
OPT_TYPE_UINT64, /* ZMQ_SNDBUF */
OPT_TYPE_UINT64, /* ZMQ_RCVBUF */
OPT_TYPE_INT64, /* ZMQ_RCVMORE */
#if VERSION_2_1
OPT_TYPE_FD, /* ZMQ_FD */
OPT_TYPE_UINT32, /* ZMQ_EVENTS */
OPT_TYPE_INT, /* ZMQ_TYPE */
OPT_TYPE_INT, /* ZMQ_LINGER */
OPT_TYPE_INT, /* ZMQ_RECONNECT_IVL */
OPT_TYPE_INT, /* ZMQ_BACKLOG */
#endif
};
#define MAX_OPTS ZMQ_BACKLOG
]],
destructor "close" {
c_call "ZMQ_Error" "zmq_close" {}
},
method "bind" {
c_call "ZMQ_Error" "zmq_bind" { "const char *", "addr" }
},
method "connect" {
c_call "ZMQ_Error" "zmq_connect" { "const char *", "addr" }
},
method "setopt" {
var_in{ "uint32_t", "opt" },
var_in{ "<any>", "val" },
var_out{ "ZMQ_Error", "err" },
c_source[[
size_t val_len;
const void *val;
socket_t fd_val;
int int_val;
uint32_t uint32_val;
uint64_t uint64_val;
int64_t int64_val;
if(${opt} > MAX_OPTS) {
return luaL_argerror(L, ${opt::idx}, "Invalid socket option.");
}
switch(opt_types[${opt}]) {
case OPT_TYPE_FD:
fd_val = luaL_checklong(L, ${val::idx});
val = &fd_val;
val_len = sizeof(fd_val);
break;
case OPT_TYPE_INT:
int_val = luaL_checklong(L, ${val::idx});
val = &int_val;
val_len = sizeof(int_val);
break;
case OPT_TYPE_UINT32:
uint32_val = luaL_checklong(L, ${val::idx});
val = &uint32_val;
val_len = sizeof(uint32_val);
break;
case OPT_TYPE_UINT64:
uint64_val = luaL_checklong(L, ${val::idx});
val = &uint64_val;
val_len = sizeof(uint64_val);
break;
case OPT_TYPE_INT64:
int64_val = luaL_checklong(L, ${val::idx});
val = &int64_val;
val_len = sizeof(int64_val);
break;
case OPT_TYPE_STR:
val = luaL_checklstring(L, ${val::idx}, &(val_len));
break;
default:
printf("Invalid socket option type, this shouldn't happen.\n");
abort();
break;
}
${err} = zmq_setsockopt(${this}, ${opt}, val, val_len);
]]
},
method "getopt" {
var_in{ "uint32_t", "opt" },
var_out{ "<any>", "val" },
var_out{ "ZMQ_Error", "err" },
c_source[[
size_t val_len;
socket_t fd_val;
int int_val;
uint32_t uint32_val;
uint64_t uint64_val;
int64_t int64_val;
#define STR_MAX 255
char str_val[STR_MAX];
if(${opt} > MAX_OPTS) {
lua_pushnil(L);
lua_pushliteral(L, "Invalid socket option.");
return 2;
}
switch(opt_types[${opt}]) {
case OPT_TYPE_FD:
val_len = sizeof(fd_val);
${err} = zmq_getsockopt(${this}, ${opt}, &fd_val, &val_len);
if(0 == ${err}) {
lua_pushinteger(L, (lua_Integer)fd_val);
return 1;
}
break;
case OPT_TYPE_INT:
val_len = sizeof(int_val);
${err} = zmq_getsockopt(${this}, ${opt}, &int_val, &val_len);
if(0 == ${err}) {
lua_pushinteger(L, (lua_Integer)int_val);
return 1;
}
break;
case OPT_TYPE_UINT32:
val_len = sizeof(uint32_val);
${err} = zmq_getsockopt(${this}, ${opt}, &uint32_val, &val_len);
if(0 == ${err}) {
lua_pushinteger(L, (lua_Integer)uint32_val);
return 1;
}
break;
case OPT_TYPE_UINT64:
val_len = sizeof(uint64_val);
${err} = zmq_getsockopt(${this}, ${opt}, &uint64_val, &val_len);
if(0 == ${err}) {
lua_pushinteger(L, (lua_Integer)uint64_val);
return 1;
}
break;
case OPT_TYPE_INT64:
val_len = sizeof(int64_val);
${err} = zmq_getsockopt(${this}, ${opt}, &int64_val, &val_len);
if(0 == ${err}) {
lua_pushinteger(L, (lua_Integer)int64_val);
return 1;
}
break;
case OPT_TYPE_STR:
val_len = STR_MAX;
${err} = zmq_getsockopt(${this}, ${opt}, str_val, &val_len);
if(0 == ${err}) {
lua_pushlstring(L, str_val, val_len);
return 1;
}
#undef STR_MAX
break;
default:
printf("Invalid socket option type, this shouldn't happen.\n");
abort();
break;
}
lua_pushnil(L);
]]
},
method "send" {
var_in{ "const char *", "data" },
var_in{ "int", "flags", is_optional = true },
var_out{ "ZMQ_Error", "err" },
c_source[[
zmq_msg_t msg;
/* initialize message */
${err} = zmq_msg_init_size(&msg, ${data}_len);
if(0 == ${err}) {
/* fill message */
memcpy(zmq_msg_data(&msg), ${data}, ${data}_len);
/* send message */
${err} = zmq_send(${this}, &msg, ${flags});
/* close message */
zmq_msg_close(&msg);
}
]]
},
method "recv" {
var_in{ "int", "flags", is_optional = true },
var_out{ "const char *", "data", has_length = true },
var_out{ "ZMQ_Error", "err" },
c_source[[
zmq_msg_t msg;
/* initialize message */
${err} = zmq_msg_init(&msg);
if(0 == ${err}) {
/* receive message */
${err} = zmq_recv(${this}, &msg, ${flags});
if(0 == ${err}) {
${data} = zmq_msg_data(&msg);
${data}_len = zmq_msg_size(&msg);
}
}
]],
c_source "post" [[
/* close message */
zmq_msg_close(&msg);
]]
},
}