/* * nixio - Linux I/O library for lua * * Copyright (C) 2009 Steven Barth <steven@midlink.org> * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "nixio.h" #include <stdio.h> #include <string.h> #include <errno.h> #include <signal.h> #define VERSION 0.4 /* pushes nil, error number and errstring on the stack */ int nixio__perror(lua_State *L) { if (errno == EAGAIN || errno == EWOULDBLOCK) { lua_pushboolean(L, 0); } else { lua_pushnil(L); } lua_pushinteger(L, errno); lua_pushstring(L, strerror(errno)); return 3; } /* pushes true, if operation succeeded, otherwise call nixio__perror */ int nixio__pstatus(lua_State *L, int condition) { if (condition) { lua_pushboolean(L, 1); return 1; } else { return nixio__perror(L); } } /* checks whether the first argument is a socket and returns it */ nixio_sock* nixio__checksock(lua_State *L) { nixio_sock *sock = (nixio_sock*)luaL_checkudata(L, 1, NIXIO_META); luaL_argcheck(L, sock->fd != -1, 1, "invalid socket object"); return sock; } /* read fd from nixio_sock object */ int nixio__checksockfd(lua_State *L) { return nixio__checksock(L)->fd; } /* return any possible fd, otherwise error out */ int nixio__checkfd(lua_State *L, int ud) { int fd = nixio__tofd(L, ud); return (fd != -1) ? fd : luaL_argerror(L, ud, "invalid file descriptor"); } /* return any possible fd */ int nixio__tofd(lua_State *L, int ud) { void *udata = lua_touserdata(L, ud); int fd = -1; if (lua_getmetatable(L, ud)) { luaL_getmetatable(L, NIXIO_META); luaL_getmetatable(L, NIXIO_FILE_META); luaL_getmetatable(L, LUA_FILEHANDLE); if (lua_rawequal(L, -3, -4)) { fd = ((nixio_sock*)udata)->fd; } else if (lua_rawequal(L, -2, -4)) { fd = *((int*)udata); } else if (lua_rawequal(L, -1, -4)) { fd = (*((FILE **)udata)) ? fileno(*((FILE **)udata)) : -1; } lua_pop(L, 4); } return fd; } /* An empty iterator */ int nixio__nulliter(lua_State *L) { lua_pushnil(L); return 1; } static int nixio_errno(lua_State *L) { lua_pushinteger(L, errno); return 1; } static int nixio_strerror(lua_State *L) { lua_pushstring(L, strerror(luaL_checkinteger(L, 1))); return 1; } /* object table */ static const luaL_reg R[] = { {"errno", nixio_errno}, {"strerror", nixio_strerror}, {NULL, NULL} }; /* entry point */ NIXIO_API int luaopen_nixio(lua_State *L) { /* create metatable */ luaL_newmetatable(L, NIXIO_META); /* metatable.__index = metatable */ lua_pushvalue(L, -1); lua_setfield(L, -2, "__index"); /* register module */ luaL_register(L, "nixio", R); /* register metatable as socket_meta */ lua_pushvalue(L, -2); lua_setfield(L, -2, "meta_socket"); /* register methods */ #ifdef __WINNT__ nixio_open__mingw(L); #endif nixio_open_file(L); nixio_open_socket(L); nixio_open_sockopt(L); nixio_open_bind(L); nixio_open_address(L); nixio_open_protoent(L); nixio_open_poll(L); nixio_open_io(L); nixio_open_splice(L); nixio_open_process(L); nixio_open_syslog(L); nixio_open_bit(L); nixio_open_bin(L); nixio_open_fs(L); nixio_open_user(L); #ifndef NO_TLS nixio_open_tls_crypto(L); nixio_open_tls_context(L); nixio_open_tls_socket(L); #endif /* module version */ lua_pushinteger(L, VERSION); lua_setfield(L, -2, "version"); /* some constants */ lua_newtable(L); lua_pushliteral(L, NIXIO_SEP); lua_setfield(L, -2, "sep"); lua_pushliteral(L, NIXIO_PATHSEP); lua_setfield(L, -2, "pathsep"); lua_pushinteger(L, NIXIO_BUFFERSIZE); lua_setfield(L, -2, "buffersize"); NIXIO_PUSH_CONSTANT(EACCES); NIXIO_PUSH_CONSTANT(EINTR); NIXIO_PUSH_CONSTANT(ENOSYS); NIXIO_PUSH_CONSTANT(EINVAL); NIXIO_PUSH_CONSTANT(EAGAIN); NIXIO_PUSH_CONSTANT(ENOMEM); NIXIO_PUSH_CONSTANT(ENOENT); NIXIO_PUSH_CONSTANT(ECHILD); NIXIO_PUSH_CONSTANT(EIO); NIXIO_PUSH_CONSTANT(EBADF); NIXIO_PUSH_CONSTANT(EFAULT); NIXIO_PUSH_CONSTANT(EFBIG); NIXIO_PUSH_CONSTANT(ENOSPC); NIXIO_PUSH_CONSTANT(EPIPE); NIXIO_PUSH_CONSTANT(ESPIPE); NIXIO_PUSH_CONSTANT(EISDIR); NIXIO_PUSH_CONSTANT(EPERM); NIXIO_PUSH_CONSTANT(EEXIST); NIXIO_PUSH_CONSTANT(EMFILE); NIXIO_PUSH_CONSTANT(ENAMETOOLONG); NIXIO_PUSH_CONSTANT(ENFILE); NIXIO_PUSH_CONSTANT(ENODEV); NIXIO_PUSH_CONSTANT(EXDEV); NIXIO_PUSH_CONSTANT(ENOTDIR); NIXIO_PUSH_CONSTANT(ENXIO); NIXIO_PUSH_CONSTANT(EROFS); NIXIO_PUSH_CONSTANT(EBUSY); NIXIO_PUSH_CONSTANT(ESRCH); NIXIO_PUSH_CONSTANT(SIGINT); NIXIO_PUSH_CONSTANT(SIGTERM); NIXIO_PUSH_CONSTANT(SIGSEGV); #ifndef __WINNT__ NIXIO_PUSH_CONSTANT(EALREADY); NIXIO_PUSH_CONSTANT(EINPROGRESS); NIXIO_PUSH_CONSTANT(EWOULDBLOCK); NIXIO_PUSH_CONSTANT(ELOOP); NIXIO_PUSH_CONSTANT(EOVERFLOW); NIXIO_PUSH_CONSTANT(ETXTBSY); NIXIO_PUSH_CONSTANT(EAFNOSUPPORT); NIXIO_PUSH_CONSTANT(ENOBUFS); NIXIO_PUSH_CONSTANT(EPROTONOSUPPORT); NIXIO_PUSH_CONSTANT(ENOPROTOOPT); NIXIO_PUSH_CONSTANT(EADDRINUSE); NIXIO_PUSH_CONSTANT(ENETDOWN); NIXIO_PUSH_CONSTANT(ENETUNREACH); NIXIO_PUSH_CONSTANT(SIGALRM); NIXIO_PUSH_CONSTANT(SIGKILL); NIXIO_PUSH_CONSTANT(SIGHUP); NIXIO_PUSH_CONSTANT(SIGSTOP); NIXIO_PUSH_CONSTANT(SIGCONT); NIXIO_PUSH_CONSTANT(SIGCHLD); NIXIO_PUSH_CONSTANT(SIGQUIT); NIXIO_PUSH_CONSTANT(SIGUSR1); NIXIO_PUSH_CONSTANT(SIGUSR2); NIXIO_PUSH_CONSTANT(SIGIO); NIXIO_PUSH_CONSTANT(SIGURG); NIXIO_PUSH_CONSTANT(SIGPIPE); lua_pushvalue(L, -1); lua_setfield(L, -3, "const_sock"); signal(SIGPIPE, SIG_IGN); #endif /* !__WINNT__ */ lua_setfield(L, -2, "const"); /* remove meta table */ lua_remove(L, -2); return 1; }