aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDiego Nehab <diego@tecgraf.puc-rio.br>2002-07-03 19:06:54 +0000
committerDiego Nehab <diego@tecgraf.puc-rio.br>2002-07-03 19:06:54 +0000
commit88026bef8a5ca586e354965a79134646fb566c72 (patch)
tree7594d9748c9dfd5db4addc0756b0cf86a0d9206a
parent9b8bce6465d2c7de84782f9e12f529a020a16444 (diff)
downloadluasocket-88026bef8a5ca586e354965a79134646fb566c72.tar.gz
luasocket-88026bef8a5ca586e354965a79134646fb566c72.tar.bz2
luasocket-88026bef8a5ca586e354965a79134646fb566c72.zip
Initial revision
-rw-r--r--src/inet.h36
-rw-r--r--src/select.h7
-rw-r--r--src/socket.h18
-rw-r--r--src/timeout.c160
-rw-r--r--src/timeout.h20
-rw-r--r--src/udp.c287
-rw-r--r--src/udp.h24
7 files changed, 552 insertions, 0 deletions
diff --git a/src/inet.h b/src/inet.h
new file mode 100644
index 0000000..3b0453e
--- /dev/null
+++ b/src/inet.h
@@ -0,0 +1,36 @@
1/*=========================================================================*\
2* Internet domain class
3* RCS ID: $Id$
4\*=========================================================================*/
5#ifndef INET_H_
6#define INET_H_
7
8#include <lua.h>
9#include "lssock.h"
10
11/* class name */
12#define INET_CLASS "luasocket(inet)"
13
14/*-------------------------------------------------------------------------*\
15* Socket fields
16\*-------------------------------------------------------------------------*/
17#define INET_FIELDS SOCK_FIELDS
18
19/*-------------------------------------------------------------------------*\
20* Socket structure
21\*-------------------------------------------------------------------------*/
22typedef t_sock t_inet;
23typedef t_inet *p_inet;
24
25/*-------------------------------------------------------------------------*\
26* Exported functions
27\*-------------------------------------------------------------------------*/
28void inet_open(lua_State *L);
29void inet_construct(lua_State *L, p_inet inet);
30void inet_inherit(lua_State *L, cchar *lsclass);
31
32cchar *inet_tryconnect(p_sock sock, cchar *address, ushort);
33cchar *inet_trybind(p_sock sock, cchar *address, ushort);
34cchar *inet_trysocket(p_inet inet, int type);
35
36#endif /* INET_H_ */
diff --git a/src/select.h b/src/select.h
new file mode 100644
index 0000000..c3267ad
--- /dev/null
+++ b/src/select.h
@@ -0,0 +1,7 @@
1#ifndef SLCT_H_
2#define SLCT_H_
3
4void select_addclass(lua_State *L, cchar *lsclass);
5void select_open(lua_State *L);
6
7#endif
diff --git a/src/socket.h b/src/socket.h
new file mode 100644
index 0000000..c9dee20
--- /dev/null
+++ b/src/socket.h
@@ -0,0 +1,18 @@
1#ifndef SOCK_H_
2#define SOCK_H_
3
4#include <lua.h>
5#include "lsfd.h"
6
7#define SOCK_CLASS "luasocket(sock)"
8
9#define SOCK_FIELDS FD_FIELDS
10
11typedef t_fd t_sock;
12typedef t_sock *p_sock;
13
14void sock_open(lua_State *L);
15void sock_construct(lua_State *L, p_sock sock);
16void sock_inherit(lua_State *L, cchar *lsclass);
17
18#endif /* SOCK_H_ */
diff --git a/src/timeout.c b/src/timeout.c
new file mode 100644
index 0000000..266a86e
--- /dev/null
+++ b/src/timeout.c
@@ -0,0 +1,160 @@
1/*=========================================================================*\
2* Timeout management functions
3\*=========================================================================*/
4#include <lua.h>
5#include <lauxlib.h>
6
7#include "lspriv.h"
8#include "lstm.h"
9
10#include <stdio.h>
11
12#ifdef WIN32
13#include <windows.h>
14#else
15#include <sys/times.h>
16#include <time.h>
17#include <unistd.h>
18#endif
19
20/*=========================================================================*\
21* Internal function prototypes
22\*=========================================================================*/
23#ifdef _DEBUG
24static int tm_lua_time(lua_State *L);
25static int tm_lua_sleep(lua_State *L);
26#endif
27
28/*=========================================================================*\
29* Exported functions.
30\*=========================================================================*/
31/*-------------------------------------------------------------------------*\
32* Sets timeout limits
33* Input
34* tm: timeout control structure
35* mode: block or return timeout
36* value: timeout value in miliseconds
37\*-------------------------------------------------------------------------*/
38void tm_set(p_tm tm, int tm_block, int tm_return)
39{
40 tm->tm_block = tm_block;
41 tm->tm_return = tm_return;
42}
43
44/*-------------------------------------------------------------------------*\
45* Returns timeout limits
46* Input
47* tm: timeout control structure
48* mode: block or return timeout
49* value: timeout value in miliseconds
50\*-------------------------------------------------------------------------*/
51void tm_get(p_tm tm, int *tm_block, int *tm_return)
52{
53 if (tm_block) *tm_block = tm->tm_block;
54 if (tm_return) *tm_return = tm->tm_return;
55}
56
57/*-------------------------------------------------------------------------*\
58* Determines how much time we have left for the current io operation
59* an IO write operation.
60* Input
61* tm: timeout control structure
62* Returns
63* the number of ms left or -1 if there is no time limit
64\*-------------------------------------------------------------------------*/
65int tm_getremaining(p_tm tm)
66{
67 /* no timeout */
68 if (tm->tm_block < 0 && tm->tm_return < 0)
69 return -1;
70 /* there is no block timeout, we use the return timeout */
71 else if (tm->tm_block < 0)
72 return MAX(tm->tm_return - tm_gettime() + tm->tm_start, 0);
73 /* there is no return timeout, we use the block timeout */
74 else if (tm->tm_return < 0)
75 return tm->tm_block;
76 /* both timeouts are specified */
77 else return MIN(tm->tm_block,
78 MAX(tm->tm_return - tm_gettime() + tm->tm_start, 0));
79}
80
81/*-------------------------------------------------------------------------*\
82* Marks the operation start time in sock structure
83* Input
84* tm: timeout control structure
85\*-------------------------------------------------------------------------*/
86void tm_markstart(p_tm tm)
87{
88 tm->tm_start = tm_gettime();
89 tm->tm_end = tm->tm_start;
90}
91
92/*-------------------------------------------------------------------------*\
93* Returns the length of the operation in ms
94* Input
95* tm: timeout control structure
96\*-------------------------------------------------------------------------*/
97int tm_getelapsed(p_tm tm)
98{
99 return tm->tm_end - tm->tm_start;
100}
101
102/*-------------------------------------------------------------------------*\
103* Gets time in ms, relative to system startup.
104* Returns
105* time in ms.
106\*-------------------------------------------------------------------------*/
107#ifdef WIN32
108int tm_gettime(void)
109{
110 return GetTickCount();
111}
112#else
113int tm_gettime(void)
114{
115 struct tms t;
116 return (times(&t)*1000)/CLK_TCK;
117}
118#endif
119
120/*-------------------------------------------------------------------------*\
121* Initializes module
122\*-------------------------------------------------------------------------*/
123void tm_open(lua_State *L)
124{
125 (void) L;
126#ifdef _DEBUG
127 lua_pushcfunction(L, tm_lua_time);
128 lua_setglobal(L, "_time");
129 lua_pushcfunction(L, tm_lua_sleep);
130 lua_setglobal(L, "_sleep");
131#endif
132}
133
134/*=========================================================================*\
135* Test support functions
136\*=========================================================================*/
137/*-------------------------------------------------------------------------*\
138* Returns the time the system has been up, in secconds.
139\*-------------------------------------------------------------------------*/
140#ifdef _DEBUG
141static int tm_lua_time(lua_State *L)
142{
143 lua_pushnumber(L, tm_gettime()/1000.0);
144 return 1;
145}
146
147/*-------------------------------------------------------------------------*\
148* Sleep for n seconds.
149\*-------------------------------------------------------------------------*/
150int tm_lua_sleep(lua_State *L)
151{
152 double n = luaL_check_number(L, 1);
153#ifdef WIN32
154 Sleep(n*1000);
155#else
156 sleep(n);
157#endif
158 return 0;
159}
160#endif
diff --git a/src/timeout.h b/src/timeout.h
new file mode 100644
index 0000000..af7e591
--- /dev/null
+++ b/src/timeout.h
@@ -0,0 +1,20 @@
1#ifndef _TM_H
2#define _TM_H
3
4typedef struct t_tm_tag {
5 int tm_return;
6 int tm_block;
7 int tm_start;
8 int tm_end;
9} t_tm;
10typedef t_tm *p_tm;
11
12void tm_set(p_tm tm, int tm_block, int tm_return);
13int tm_getremaining(p_tm tm);
14int tm_getelapsed(p_tm tm);
15int tm_gettime(void);
16void tm_get(p_tm tm, int *tm_block, int *tm_return);
17void tm_markstart(p_tm tm);
18void tm_open(lua_State *L);
19
20#endif
diff --git a/src/udp.c b/src/udp.c
new file mode 100644
index 0000000..0dc0df8
--- /dev/null
+++ b/src/udp.c
@@ -0,0 +1,287 @@
1/*=========================================================================*\
2* UDP socket object implementation (inherits from sock and inet)
3\*=========================================================================*/
4#include <string.h>
5
6#include <lua.h>
7#include <lauxlib.h>
8
9#include "lsinet.h"
10#include "lsudp.h"
11#include "lscompat.h"
12#include "lsselect.h"
13
14/*=========================================================================*\
15* Internal function prototypes.
16\*=========================================================================*/
17static int udp_lua_send(lua_State *L);
18static int udp_lua_sendto(lua_State *L);
19static int udp_lua_receive(lua_State *L);
20static int udp_lua_receivefrom(lua_State *L);
21static int udp_lua_setpeername(lua_State *L);
22static int udp_lua_setsockname(lua_State *L);
23
24static int udp_global_udpsocket(lua_State *L);
25
26static struct luaL_reg funcs[] = {
27 {"send", udp_lua_send},
28 {"sendto", udp_lua_sendto},
29 {"receive", udp_lua_receive},
30 {"receivefrom", udp_lua_receivefrom},
31 {"setpeername", udp_lua_setpeername},
32 {"setsockname", udp_lua_setsockname},
33};
34
35/*=========================================================================*\
36* Exported functions
37\*=========================================================================*/
38/*-------------------------------------------------------------------------*\
39* Initializes module
40\*-------------------------------------------------------------------------*/
41void udp_open(lua_State *L)
42{
43 unsigned int i;
44 priv_newclass(L, UDP_CLASS);
45 udp_inherit(L, UDP_CLASS);
46 /* declare global functions */
47 lua_pushcfunction(L, udp_global_udpsocket);
48 lua_setglobal(L, "udpsocket");
49 for (i = 0; i < sizeof(funcs)/sizeof(funcs[0]); i++)
50 priv_newglobalmethod(L, funcs[i].name);
51 /* make class selectable */
52 select_addclass(L, UDP_CLASS);
53}
54
55/*-------------------------------------------------------------------------*\
56* Hook object methods to methods table.
57\*-------------------------------------------------------------------------*/
58void udp_inherit(lua_State *L, cchar *lsclass)
59{
60 unsigned int i;
61 inet_inherit(L, lsclass);
62 for (i = 0; i < sizeof(funcs)/sizeof(funcs[0]); i++) {
63 lua_pushcfunction(L, funcs[i].func);
64 priv_setmethod(L, lsclass, funcs[i].name);
65 }
66}
67
68/*-------------------------------------------------------------------------*\
69* Initializes socket structure
70\*-------------------------------------------------------------------------*/
71void udp_construct(lua_State *L, p_udp udp)
72{
73 inet_construct(L, (p_inet) udp);
74 udp->udp_connected = 0;
75}
76
77/*-------------------------------------------------------------------------*\
78* Creates a socket structure and initializes it. A socket object is
79* left in the Lua stack.
80* Returns
81* pointer to allocated structure
82\*-------------------------------------------------------------------------*/
83p_udp udp_push(lua_State *L)
84{
85 p_udp udp = (p_udp) lua_newuserdata(L, sizeof(t_udp));
86 priv_setclass(L, UDP_CLASS);
87 udp_construct(L, udp);
88 return udp;
89}
90
91/*=========================================================================*\
92* Socket table constructors
93\*=========================================================================*/
94/*-------------------------------------------------------------------------*\
95* Creates a udp socket object and returns it to the Lua script.
96* Lua Input: [options]
97* options: socket options table
98* Lua Returns
99* On success: udp socket
100* On error: nil, followed by an error message
101\*-------------------------------------------------------------------------*/
102static int udp_global_udpsocket(lua_State *L)
103{
104 int oldtop = lua_gettop(L);
105 p_udp udp = udp_push(L);
106 cchar *err = inet_trysocket((p_inet) udp, SOCK_DGRAM);
107 if (err) {
108 lua_pushnil(L);
109 lua_pushstring(L, err);
110 return 2;
111 }
112 if (oldtop < 1) return 1;
113 err = compat_trysetoptions(L, udp->fd);
114 if (err) {
115 lua_pushnil(L);
116 lua_pushstring(L, err);
117 return 2;
118 }
119 return 1;
120}
121
122/*=========================================================================*\
123* Socket table methods
124\*=========================================================================*/
125/*-------------------------------------------------------------------------*\
126* Receives data from a UDP socket
127* Lua Input: sock [, wanted]
128* sock: client socket created by the connect function
129* wanted: the number of bytes expected (default: LUASOCKET_UDPBUFFERSIZE)
130* Lua Returns
131* On success: datagram received
132* On error: nil, followed by an error message
133\*-------------------------------------------------------------------------*/
134static int udp_lua_receive(lua_State *L)
135{
136 p_udp udp = (p_udp) lua_touserdata(L, 1);
137 unsigned char buffer[UDP_DATAGRAMSIZE];
138 size_t got, wanted = (size_t) luaL_opt_number(L, 2, sizeof(buffer));
139 int err;
140 p_tm tm = &udp->base_tm;
141 wanted = MIN(wanted, sizeof(buffer));
142 tm_markstart(tm);
143 err = compat_recv(udp->fd, buffer, wanted, &got, tm_getremaining(tm));
144 if (err == PRIV_CLOSED) err = PRIV_REFUSED;
145 if (err != PRIV_DONE) lua_pushnil(L);
146 else lua_pushlstring(L, buffer, got);
147 priv_pusherror(L, err);
148 return 2;
149}
150
151/*-------------------------------------------------------------------------*\
152* Receives a datagram from a UDP socket
153* Lua Input: sock [, wanted]
154* sock: client socket created by the connect function
155* wanted: the number of bytes expected (default: LUASOCKET_UDPBUFFERSIZE)
156* Lua Returns
157* On success: datagram received, ip and port of sender
158* On error: nil, followed by an error message
159\*-------------------------------------------------------------------------*/
160static int udp_lua_receivefrom(lua_State *L)
161{
162 p_udp udp = (p_udp) lua_touserdata(L, 1);
163 p_tm tm = &udp->base_tm;
164 struct sockaddr_in peer;
165 size_t peer_len = sizeof(peer);
166 unsigned char buffer[UDP_DATAGRAMSIZE];
167 size_t wanted = (size_t) luaL_opt_number(L, 2, sizeof(buffer));
168 size_t got;
169 int err;
170 if (udp->udp_connected) lua_error(L, "receivefrom on connected socket");
171 tm_markstart(tm);
172 wanted = MIN(wanted, sizeof(buffer));
173 err = compat_recvfrom(udp->fd, buffer, wanted, &got, tm_getremaining(tm),
174 (SA *) &peer, &peer_len);
175 if (err == PRIV_CLOSED) err = PRIV_REFUSED;
176 if (err == PRIV_DONE) {
177 lua_pushlstring(L, buffer, got);
178 lua_pushstring(L, inet_ntoa(peer.sin_addr));
179 lua_pushnumber(L, ntohs(peer.sin_port));
180 return 3;
181 } else {
182 lua_pushnil(L);
183 priv_pusherror(L, err);
184 return 2;
185 }
186}
187
188/*-------------------------------------------------------------------------*\
189* Send data through a connected UDP socket
190* Lua Input: sock, data
191* sock: udp socket
192* data: data to be sent
193* Lua Returns
194* On success: nil, followed by the total number of bytes sent
195* On error: error message
196\*-------------------------------------------------------------------------*/
197static int udp_lua_send(lua_State *L)
198{
199 p_udp udp = (p_udp) lua_touserdata(L, 1);
200 p_tm tm = &udp->base_tm;
201 size_t wanted, sent = 0;
202 int err;
203 cchar *data = luaL_check_lstr(L, 2, &wanted);
204 if (!udp->udp_connected) lua_error(L, "send on unconnected socket");
205 tm_markstart(tm);
206 err = compat_send(udp->fd, data, wanted, &sent, tm_getremaining(tm));
207 priv_pusherror(L, err == PRIV_CLOSED ? PRIV_REFUSED : err);
208 lua_pushnumber(L, sent);
209 return 2;
210}
211
212/*-------------------------------------------------------------------------*\
213* Send data through a unconnected UDP socket
214* Lua Input: sock, data, ip, port
215* sock: udp socket
216* data: data to be sent
217* ip: ip address of target
218* port: port in target
219* Lua Returns
220* On success: nil, followed by the total number of bytes sent
221* On error: error message
222\*-------------------------------------------------------------------------*/
223static int udp_lua_sendto(lua_State *L)
224{
225 p_udp udp = (p_udp) lua_touserdata(L, 1);
226 size_t wanted, sent = 0;
227 cchar *data = luaL_check_lstr(L, 2, &wanted);
228 cchar *ip = luaL_check_string(L, 3);
229 ushort port = (ushort) luaL_check_number(L, 4);
230 p_tm tm = &udp->base_tm;
231 struct sockaddr_in peer;
232 int err;
233 if (udp->udp_connected) lua_error(L, "sendto on connected socket");
234 memset(&peer, 0, sizeof(peer));
235 if (!inet_aton(ip, &peer.sin_addr)) lua_error(L, "invalid ip address");
236 peer.sin_family = AF_INET;
237 peer.sin_port = htons(port);
238 tm_markstart(tm);
239 err = compat_sendto(udp->fd, data, wanted, &sent, tm_getremaining(tm),
240 (SA *) &peer, sizeof(peer));
241 priv_pusherror(L, err == PRIV_CLOSED ? PRIV_REFUSED : err);
242 lua_pushnumber(L, sent);
243 return 2;
244}
245
246/*-------------------------------------------------------------------------*\
247* Associates a local address to an UDP socket
248* Lua Input: address, port
249* address: host name or ip address to bind to
250* port: port to bind to
251* Lua Returns
252* On success: nil
253* On error: error message
254\*-------------------------------------------------------------------------*/
255static int udp_lua_setsockname(lua_State * L)
256{
257 p_udp udp = (p_udp) lua_touserdata(L, 1);
258 cchar *address = luaL_check_string(L, 2);
259 ushort port = (ushort) luaL_check_number(L, 3);
260 cchar *err = inet_trybind((p_inet) udp, address, port);
261 if (err) lua_pushstring(L, err);
262 else lua_pushnil(L);
263 return 1;
264}
265
266/*-------------------------------------------------------------------------*\
267* Sets a peer for a UDP socket
268* Lua Input: address, port
269* address: remote host name
270* port: remote host port
271* Lua Returns
272* On success: nil
273* On error: error message
274\*-------------------------------------------------------------------------*/
275static int udp_lua_setpeername(lua_State *L)
276{
277 p_udp udp = (p_udp) lua_touserdata(L, 1);
278 cchar *address = luaL_check_string(L, 2);
279 ushort port = (ushort) luaL_check_number(L, 3);
280 cchar *err = inet_tryconnect((p_inet) udp, address, port);
281 if (!err) {
282 udp->udp_connected = 1;
283 lua_pushnil(L);
284 } else lua_pushstring(L, err);
285 return 1;
286}
287
diff --git a/src/udp.h b/src/udp.h
new file mode 100644
index 0000000..3c82c29
--- /dev/null
+++ b/src/udp.h
@@ -0,0 +1,24 @@
1#ifndef UDP_H_
2#define UDP_H_
3
4#include "lsinet.h"
5
6#define UDP_CLASS "luasocket(UDP socket)"
7
8#define UDP_DATAGRAMSIZE 576
9
10#define UDP_FIELDS \
11 INET_FIELDS; \
12 int udp_connected
13
14typedef struct t_udp_tag {
15 UDP_FIELDS;
16} t_udp;
17typedef t_udp *p_udp;
18
19void udp_inherit(lua_State *L, cchar *lsclass);
20void udp_construct(lua_State *L, p_udp udp);
21void udp_open(lua_State *L);
22p_udp udp_push(lua_State *L);
23
24#endif