flm01/mote/v1/openwrt/package/luci/libs/nixio/src/nixio.c

238 lines
5.8 KiB
C

/*
* 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.3
/* 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_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(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;
}