184 lines
3.9 KiB
C
184 lines
3.9 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 <errno.h>
|
||
|
#include <string.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <stdio.h>
|
||
|
#include <sys/types.h>
|
||
|
|
||
|
|
||
|
/**
|
||
|
* send() / sendto() helper
|
||
|
*/
|
||
|
static int nixio_sock__sendto(lua_State *L, int to) {
|
||
|
nixio_sock *sock = nixio__checksock(L);
|
||
|
struct sockaddr *addr = NULL;
|
||
|
socklen_t alen = 0;
|
||
|
int argoff = 2;
|
||
|
|
||
|
if (to) {
|
||
|
argoff += 2;
|
||
|
const char *address = luaL_checkstring(L, 3);
|
||
|
struct sockaddr_storage addrstor;
|
||
|
addr = (struct sockaddr*)&addrstor;
|
||
|
|
||
|
nixio_addr naddr;
|
||
|
memset(&naddr, 0, sizeof(naddr));
|
||
|
strncpy(naddr.host, address, sizeof(naddr.host) - 1);
|
||
|
naddr.port = (uint16_t)luaL_checkinteger(L, 4);
|
||
|
naddr.family = sock->domain;
|
||
|
|
||
|
if (nixio__addr_write(&naddr, addr)) {
|
||
|
return nixio__perror_s(L);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
size_t len;
|
||
|
ssize_t sent;
|
||
|
const char *data = luaL_checklstring(L, 2, &len);
|
||
|
|
||
|
if (lua_gettop(L) > argoff) {
|
||
|
int offset = luaL_optint(L, argoff + 1, 0);
|
||
|
if (offset) {
|
||
|
if (offset < len) {
|
||
|
data += offset;
|
||
|
len -= offset;
|
||
|
} else {
|
||
|
len = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
unsigned int wlen = luaL_optint(L, argoff + 2, len);
|
||
|
if (wlen < len) {
|
||
|
len = wlen;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
do {
|
||
|
sent = sendto(sock->fd, data, len, 0, addr, alen);
|
||
|
} while(sent == -1 && errno == EINTR);
|
||
|
if (sent >= 0) {
|
||
|
lua_pushinteger(L, sent);
|
||
|
return 1;
|
||
|
} else {
|
||
|
return nixio__perror_s(L);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* send(data)
|
||
|
*/
|
||
|
static int nixio_sock_send(lua_State *L) {
|
||
|
return nixio_sock__sendto(L, 0);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* sendto(data, address, port)
|
||
|
*/
|
||
|
static int nixio_sock_sendto(lua_State *L) {
|
||
|
return nixio_sock__sendto(L, 1);
|
||
|
}
|
||
|
|
||
|
|
||
|
/**
|
||
|
* recv() / recvfrom() helper
|
||
|
*/
|
||
|
static int nixio_sock__recvfrom(lua_State *L, int from) {
|
||
|
nixio_sock *sock = nixio__checksock(L);
|
||
|
char buffer[NIXIO_BUFFERSIZE];
|
||
|
struct sockaddr_storage addrobj;
|
||
|
uint req = luaL_checkinteger(L, 2);
|
||
|
int readc;
|
||
|
|
||
|
if (from && sock->domain != AF_INET && sock->domain != AF_INET6) {
|
||
|
return luaL_argerror(L, 1, "supported families: inet, inet6");
|
||
|
}
|
||
|
|
||
|
struct sockaddr *addr = (from) ? (struct sockaddr*)&addrobj : NULL;
|
||
|
socklen_t alen = (from) ? sizeof(addrobj) : 0;
|
||
|
|
||
|
/* We limit the readsize to NIXIO_BUFFERSIZE */
|
||
|
req = (req > NIXIO_BUFFERSIZE) ? NIXIO_BUFFERSIZE : req;
|
||
|
|
||
|
do {
|
||
|
readc = recvfrom(sock->fd, buffer, req, 0, addr, &alen);
|
||
|
} while (readc == -1 && errno == EINTR);
|
||
|
|
||
|
#ifdef __WINNT__
|
||
|
if (readc < 0) {
|
||
|
int e = WSAGetLastError();
|
||
|
if (e == WSAECONNRESET || e == WSAECONNABORTED || e == WSAESHUTDOWN) {
|
||
|
readc = 0;
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
if (readc < 0) {
|
||
|
return nixio__perror_s(L);
|
||
|
} else {
|
||
|
lua_pushlstring(L, buffer, readc);
|
||
|
|
||
|
if (!from) {
|
||
|
return 1;
|
||
|
} else {
|
||
|
nixio_addr naddr;
|
||
|
if (!nixio__addr_parse(&naddr, (struct sockaddr *)&addrobj)) {
|
||
|
lua_pushstring(L, naddr.host);
|
||
|
lua_pushinteger(L, naddr.port);
|
||
|
return 3;
|
||
|
} else {
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* recv(count)
|
||
|
*/
|
||
|
static int nixio_sock_recv(lua_State *L) {
|
||
|
return nixio_sock__recvfrom(L, 0);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* recvfrom(count)
|
||
|
*/
|
||
|
static int nixio_sock_recvfrom(lua_State *L) {
|
||
|
return nixio_sock__recvfrom(L, 1);
|
||
|
}
|
||
|
|
||
|
|
||
|
/* module table */
|
||
|
static const luaL_reg M[] = {
|
||
|
{"send", nixio_sock_send},
|
||
|
{"sendto", nixio_sock_sendto},
|
||
|
{"recv", nixio_sock_recv},
|
||
|
{"recvfrom",nixio_sock_recvfrom},
|
||
|
{"write", nixio_sock_send},
|
||
|
{"read", nixio_sock_recv},
|
||
|
{NULL, NULL}
|
||
|
};
|
||
|
|
||
|
void nixio_open_io(lua_State *L) {
|
||
|
lua_pushvalue(L, -2);
|
||
|
luaL_register(L, NULL, M);
|
||
|
lua_pop(L, 1);
|
||
|
}
|