]> git.saurik.com Git - wxWidgets.git/blame - src/unix/gsocket.c
Set the selection under Motif in wxChoiceDialog
[wxWidgets.git] / src / unix / gsocket.c
CommitLineData
a324a7bc
GL
1/* -------------------------------------------------------------------------
2 * Project: GSocket (Generic Socket) for WX
3 * Name: gsocket.c
9bf10d6b
GRG
4 * Authors: Guilhem Lavaux,
5 * Guillermo Rodriguez Garcia <guille@iies.es> (maintainer)
a324a7bc
GL
6 * Purpose: GSocket main Unix file
7 * CVSID: $Id$
a324a7bc
GL
8 * -------------------------------------------------------------------------
9 */
10
9bf10d6b
GRG
11/*
12 * PLEASE don't put C++ comments here - this is a C source file.
13 */
14
15#ifndef __GSOCKET_STANDALONE__
813c20a6 16#include "wx/setup.h"
9bf10d6b 17#endif
813c20a6 18
9bf10d6b 19#if wxUSE_SOCKETS || defined(__GSOCKET_STANDALONE__)
813c20a6 20
a324a7bc 21#include <assert.h>
efee48a0 22#include <sys/types.h>
6a2e7225
VZ
23#include <netdb.h>
24#include <sys/ioctl.h>
9bf10d6b 25
af9d1662 26#ifdef __VMS__
a324a7bc 27#include <socket.h>
5c9eff30
GRG
28struct sockaddr_un
29{
9bf10d6b
GRG
30 u_char sun_len; /* sockaddr len including null */
31 u_char sun_family; /* AF_UNIX */
32 char sun_path[108]; /* path name (gag) */
af9d1662 33};
a324a7bc
GL
34#else
35#include <sys/socket.h>
a324a7bc 36#include <sys/un.h>
af9d1662 37#endif
9bf10d6b 38
a324a7bc
GL
39#include <sys/time.h>
40#include <netinet/in.h>
41#include <arpa/inet.h>
9592b687 42#include <errno.h>
a324a7bc
GL
43#include <string.h>
44#include <unistd.h>
45#include <stdio.h>
46#include <stdlib.h>
552e8bf8 47#include <stddef.h>
9b61f868 48#include <ctype.h>
a324a7bc 49#ifdef sun
9bf10d6b 50# include <sys/filio.h>
a324a7bc 51#endif
a324a7bc 52#ifdef sgi
9bf10d6b 53# include <bstring.h>
a324a7bc 54#endif
a324a7bc 55#include <signal.h>
5a96d2f4 56
efee48a0 57#ifndef SOCKLEN_T
5a96d2f4 58
85806dc2
GRG
59#ifdef VMS
60# define SOCKLEN_T unsigned int
5a96d2f4 61#else
85806dc2
GRG
62# ifdef __GLIBC__
63# if __GLIBC__ == 2
aac67d4c 64# define SOCKLEN_T socklen_t
85806dc2
GRG
65# endif
66# else
67# define SOCKLEN_T int
68# endif
5a96d2f4
GL
69#endif
70
ddb56b38 71#endif /* SOCKLEN_T */
efee48a0 72
aac67d4c
GRG
73/*
74 * MSW defines this, Unices don't.
75 */
76#ifndef INVALID_SOCKET
77#define INVALID_SOCKET -1
78#endif
9bf10d6b
GRG
79
80/*
81 * INADDR_BROADCAST is identical to INADDR_NONE which is not defined
aac67d4c 82 * on all systems. INADDR_BROADCAST should be fine to indicate an error.
9bf10d6b
GRG
83 */
84#ifndef INADDR_NONE
85#define INADDR_NONE INADDR_BROADCAST
86#endif
87
483249fc
GRG
88#define MASK_SIGNAL() \
89{ \
90 void (*old_handler)(int); \
91 \
aa6d9706
GL
92 old_handler = signal(SIGPIPE, SIG_IGN);
93
483249fc
GRG
94#define UNMASK_SIGNAL() \
95 signal(SIGPIPE, old_handler); \
aa6d9706
GL
96}
97
54e575f9 98
9bf10d6b 99#ifndef __GSOCKET_STANDALONE__
8e907a13
VZ
100# include "wx/unix/gsockunx.h"
101# include "wx/gsocket.h"
9bf10d6b 102#else
8e907a13
VZ
103# include "gsockunx.h"
104# include "gsocket.h"
9bf10d6b
GRG
105#endif /* __GSOCKET_STANDALONE__ */
106
52a07708
VZ
107/* redefine some GUI-only functions to do nothing in console mode */
108#if defined(wxUSE_GUI) && !wxUSE_GUI
8e907a13
VZ
109# define _GSocket_GUI_Init(socket) (1)
110# define _GSocket_GUI_Destroy(socket)
111# define _GSocket_Enable_Events(socket)
112# define _GSocket_Disable_Events(socket)
113# define _GSocket_Install_Callback(socket, event)
114# define _GSocket_Uninstall_Callback(socket, event)
72041cec 115#endif /* wxUSE_GUI */
9bf10d6b 116
8e907a13
VZ
117/* debugging helpers */
118#ifdef __GSOCKET_DEBUG__
032d5581 119# define GSocket_Debug(args) printf args
8e907a13 120#else
032d5581
GRG
121# define GSocket_Debug(args)
122#endif /* __GSOCKET_DEBUG__ */
8e907a13 123
a58d5df4
GL
124/* Global initialisers */
125
5c9eff30 126int GSocket_Init(void)
a58d5df4 127{
5c9eff30 128 return 1;
a58d5df4
GL
129}
130
da051b23 131void GSocket_Cleanup(void)
a58d5df4
GL
132{
133}
134
9bf10d6b 135/* Constructors / Destructors for GSocket */
a324a7bc 136
da051b23 137GSocket *GSocket_new(void)
a324a7bc 138{
8e907a13 139 int i, success;
a324a7bc
GL
140 GSocket *socket;
141
142 socket = (GSocket *)malloc(sizeof(GSocket));
143
e00f35bb
GL
144 if (socket == NULL)
145 return NULL;
146
aac67d4c 147 socket->m_fd = INVALID_SOCKET;
483249fc
GRG
148 for (i=0;i<GSOCK_MAX_EVENT;i++)
149 {
98781fa3 150 socket->m_cbacks[i] = NULL;
a324a7bc 151 }
483249fc 152 socket->m_detected = 0;
a324a7bc
GL
153 socket->m_local = NULL;
154 socket->m_peer = NULL;
155 socket->m_error = GSOCK_NOERROR;
ef25e638
GRG
156 socket->m_server = FALSE;
157 socket->m_stream = TRUE;
158 socket->m_gui_dependent = NULL;
159 socket->m_non_blocking = FALSE;
39b91eca 160 socket->m_timeout = 10*60*1000;
ef25e638 161 /* 10 minutes * 60 sec * 1000 millisec */
aa6d9706 162 socket->m_establishing = FALSE;
a324a7bc 163
da051b23 164 /* Per-socket GUI-specific initialization */
8e907a13
VZ
165 success = _GSocket_GUI_Init(socket);
166 if (!success)
da051b23
GRG
167 {
168 free(socket);
169 return NULL;
170 }
a324a7bc
GL
171
172 return socket;
173}
174
007c77ab
RL
175void GSocket_close(GSocket *socket)
176{
177 _GSocket_Disable_Events(socket);
178 close(socket->m_fd);
179 socket->m_fd = INVALID_SOCKET;
180}
181
a324a7bc
GL
182void GSocket_destroy(GSocket *socket)
183{
184 assert(socket != NULL);
185
9bf10d6b 186 /* Check that the socket is really shutdowned */
aac67d4c 187 if (socket->m_fd != INVALID_SOCKET)
a324a7bc
GL
188 GSocket_Shutdown(socket);
189
f6f13cd4
GRG
190 /* Per-socket GUI-specific cleanup */
191 _GSocket_GUI_Destroy(socket);
192
9bf10d6b 193 /* Destroy private addresses */
a324a7bc
GL
194 if (socket->m_local)
195 GAddress_destroy(socket->m_local);
196
197 if (socket->m_peer)
198 GAddress_destroy(socket->m_peer);
199
9bf10d6b 200 /* Destroy the socket itself */
a324a7bc
GL
201 free(socket);
202}
203
9bf10d6b
GRG
204/* GSocket_Shutdown:
205 * Disallow further read/write operations on this socket, close
206 * the fd and disable all callbacks.
207 */
a324a7bc
GL
208void GSocket_Shutdown(GSocket *socket)
209{
210 int evt;
211
212 assert(socket != NULL);
213
9bf10d6b 214 /* If socket has been created, shutdown it */
aac67d4c 215 if (socket->m_fd != INVALID_SOCKET)
483249fc 216 {
533c7161 217 shutdown(socket->m_fd, 2);
007c77ab 218 GSocket_close(socket);
a324a7bc
GL
219 }
220
9bf10d6b 221 /* Disable GUI callbacks */
483249fc
GRG
222 for (evt = 0; evt < GSOCK_MAX_EVENT; evt++)
223 socket->m_cbacks[evt] = NULL;
224
557e7011 225 socket->m_detected = GSOCK_LOST_FLAG;
a324a7bc
GL
226}
227
228/* Address handling */
229
9bf10d6b
GRG
230/* GSocket_SetLocal:
231 * GSocket_GetLocal:
232 * GSocket_SetPeer:
233 * GSocket_GetPeer:
234 * Set or get the local or peer address for this socket. The 'set'
235 * functions return GSOCK_NOERROR on success, an error code otherwise.
236 * The 'get' functions return a pointer to a GAddress object on success,
237 * or NULL otherwise, in which case they set the error code of the
238 * corresponding GSocket.
239 *
240 * Error codes:
241 * GSOCK_INVSOCK - the socket is not valid.
242 * GSOCK_INVADDR - the address is not valid.
243 */
a324a7bc
GL
244GSocketError GSocket_SetLocal(GSocket *socket, GAddress *address)
245{
5a96d2f4
GL
246 assert(socket != NULL);
247
9bf10d6b 248 /* the socket must be initialized, or it must be a server */
aac67d4c 249 if ((socket->m_fd != INVALID_SOCKET && !socket->m_server))
9bf10d6b 250 {
31989b0b 251 socket->m_error = GSOCK_INVSOCK;
a324a7bc 252 return GSOCK_INVSOCK;
31989b0b 253 }
a324a7bc 254
9bf10d6b
GRG
255 /* check address */
256 if (address == NULL || address->m_family == GSOCK_NOFAMILY)
257 {
31989b0b 258 socket->m_error = GSOCK_INVADDR;
a324a7bc 259 return GSOCK_INVADDR;
31989b0b 260 }
a324a7bc
GL
261
262 if (socket->m_local)
263 GAddress_destroy(socket->m_local);
264
265 socket->m_local = GAddress_copy(address);
266
267 return GSOCK_NOERROR;
268}
269
270GSocketError GSocket_SetPeer(GSocket *socket, GAddress *address)
271{
5a96d2f4 272 assert(socket != NULL);
a324a7bc 273
9bf10d6b
GRG
274 /* check address */
275 if (address == NULL || address->m_family == GSOCK_NOFAMILY)
276 {
a324a7bc
GL
277 socket->m_error = GSOCK_INVADDR;
278 return GSOCK_INVADDR;
279 }
280
281 if (socket->m_peer)
282 GAddress_destroy(socket->m_peer);
283
284 socket->m_peer = GAddress_copy(address);
285
286 return GSOCK_NOERROR;
287}
288
289GAddress *GSocket_GetLocal(GSocket *socket)
290{
291 GAddress *address;
292 struct sockaddr addr;
85806dc2 293 SOCKLEN_T size = sizeof(addr);
aa6d9706 294 GSocketError err;
a324a7bc
GL
295
296 assert(socket != NULL);
297
9bf10d6b 298 /* try to get it from the m_local var first */
a324a7bc
GL
299 if (socket->m_local)
300 return GAddress_copy(socket->m_local);
301
9bf10d6b 302 /* else, if the socket is initialized, try getsockname */
aac67d4c 303 if (socket->m_fd == INVALID_SOCKET)
9bf10d6b 304 {
a324a7bc
GL
305 socket->m_error = GSOCK_INVSOCK;
306 return NULL;
307 }
308
9bf10d6b
GRG
309 if (getsockname(socket->m_fd, &addr, (SOCKLEN_T *) &size) < 0)
310 {
a324a7bc
GL
311 socket->m_error = GSOCK_IOERR;
312 return NULL;
313 }
314
9bf10d6b 315 /* got a valid address from getsockname, create a GAddress object */
a324a7bc 316 address = GAddress_new();
9bf10d6b
GRG
317 if (address == NULL)
318 {
e00f35bb
GL
319 socket->m_error = GSOCK_MEMERR;
320 return NULL;
321 }
9bf10d6b
GRG
322
323 err = _GAddress_translate_from(address, &addr, size);
324 if (err != GSOCK_NOERROR)
325 {
e00f35bb 326 GAddress_destroy(address);
85806dc2 327 socket->m_error = err;
e00f35bb
GL
328 return NULL;
329 }
a324a7bc
GL
330
331 return address;
332}
333
334GAddress *GSocket_GetPeer(GSocket *socket)
335{
336 assert(socket != NULL);
337
9bf10d6b 338 /* try to get it from the m_peer var */
a324a7bc
GL
339 if (socket->m_peer)
340 return GAddress_copy(socket->m_peer);
341
342 return NULL;
343}
344
345/* Server specific parts */
346
2f7c2af5 347/* GSocket_SetServer:
9bf10d6b
GRG
348 * Sets up this socket as a server. The local address must have been
349 * set with GSocket_SetLocal() before GSocket_SetServer() is called.
350 * Returns GSOCK_NOERROR on success, one of the following otherwise:
351 *
352 * Error codes:
353 * GSOCK_INVSOCK - the socket is in use.
354 * GSOCK_INVADDR - the local address has not been set.
355 * GSOCK_IOERR - low-level error.
2f7c2af5 356 */
a324a7bc
GL
357GSocketError GSocket_SetServer(GSocket *sck)
358{
483249fc 359 int arg = 1;
a324a7bc
GL
360
361 assert(sck != NULL);
362
9bf10d6b 363 /* must not be in use */
aac67d4c 364 if (sck->m_fd != INVALID_SOCKET)
9bf10d6b 365 {
a324a7bc
GL
366 sck->m_error = GSOCK_INVSOCK;
367 return GSOCK_INVSOCK;
368 }
369
9bf10d6b
GRG
370 /* the local addr must have been set */
371 if (!sck->m_local)
372 {
a324a7bc
GL
373 sck->m_error = GSOCK_INVADDR;
374 return GSOCK_INVADDR;
375 }
376
9bf10d6b
GRG
377 /* Initialize all fields */
378 sck->m_stream = TRUE;
379 sck->m_server = TRUE;
380 sck->m_oriented = TRUE;
a324a7bc 381
f61815af
GL
382 /* Create the socket */
383 sck->m_fd = socket(sck->m_local->m_realfamily, SOCK_STREAM, 0);
a324a7bc 384
aac67d4c 385 if (sck->m_fd == INVALID_SOCKET)
9bf10d6b 386 {
a324a7bc
GL
387 sck->m_error = GSOCK_IOERR;
388 return GSOCK_IOERR;
389 }
390
483249fc
GRG
391 ioctl(sck->m_fd, FIONBIO, &arg);
392 _GSocket_Enable_Events(sck);
393
9bf10d6b
GRG
394 /* Bind to the local address,
395 * retrieve the actual address bound,
396 * and listen up to 5 connections.
397 */
398 if ((bind(sck->m_fd, sck->m_local->m_addr, sck->m_local->m_len) != 0) ||
399 (getsockname(sck->m_fd,
400 sck->m_local->m_addr,
401 (SOCKLEN_T *) &sck->m_local->m_len) != 0) ||
402 (listen(sck->m_fd, 5) != 0))
403 {
007c77ab 404 GSocket_close(sck);
a324a7bc
GL
405 sck->m_error = GSOCK_IOERR;
406 return GSOCK_IOERR;
407 }
408
a324a7bc 409 return GSOCK_NOERROR;
a324a7bc
GL
410}
411
2f7c2af5 412/* GSocket_WaitConnection:
9bf10d6b
GRG
413 * Waits for an incoming client connection. Returns a pointer to
414 * a GSocket object, or NULL if there was an error, in which case
415 * the last error field will be updated for the calling GSocket.
416 *
417 * Error codes (set in the calling GSocket)
418 * GSOCK_INVSOCK - the socket is not valid or not a server.
419 * GSOCK_TIMEDOUT - timeout, no incoming connections.
420 * GSOCK_WOULDBLOCK - the call would block and the socket is nonblocking.
421 * GSOCK_MEMERR - couldn't allocate memory.
422 * GSOCK_IOERR - low-level error.
2f7c2af5 423 */
a324a7bc
GL
424GSocket *GSocket_WaitConnection(GSocket *socket)
425{
85806dc2
GRG
426 struct sockaddr from;
427 SOCKLEN_T fromlen = sizeof(from);
a324a7bc 428 GSocket *connection;
bd046d42 429 GSocketError err;
483249fc 430 int arg = 1;
a324a7bc
GL
431
432 assert(socket != NULL);
433
483249fc
GRG
434 /* Reenable CONNECTION events */
435 _GSocket_Enable(socket, GSOCK_CONNECTION);
436
f61815af 437 /* If the socket has already been created, we exit immediately */
aac67d4c 438 if (socket->m_fd == INVALID_SOCKET || !socket->m_server)
483249fc 439 {
a324a7bc
GL
440 socket->m_error = GSOCK_INVSOCK;
441 return NULL;
442 }
443
f61815af 444 /* Create a GSocket object for the new connection */
a324a7bc 445 connection = GSocket_new();
9bf10d6b 446
483249fc
GRG
447 if (!connection)
448 {
85806dc2 449 socket->m_error = GSOCK_MEMERR;
aa6d9706
GL
450 return NULL;
451 }
a324a7bc 452
9bf10d6b 453 /* Wait for a connection (with timeout) */
483249fc
GRG
454 if (_GSocket_Input_Timeout(socket) == GSOCK_TIMEDOUT)
455 {
456 GSocket_destroy(connection);
457 /* socket->m_error set by _GSocket_Input_Timeout */
458 return NULL;
459 }
460
85806dc2 461 connection->m_fd = accept(socket->m_fd, &from, (SOCKLEN_T *) &fromlen);
483249fc 462
aac67d4c 463 if (connection->m_fd == INVALID_SOCKET)
483249fc
GRG
464 {
465 if (errno == EWOULDBLOCK)
466 socket->m_error = GSOCK_WOULDBLOCK;
467 else
468 socket->m_error = GSOCK_IOERR;
469
a324a7bc 470 GSocket_destroy(connection);
a324a7bc
GL
471 return NULL;
472 }
473
f61815af 474 /* Initialize all fields */
a324a7bc 475 connection->m_server = FALSE;
483249fc 476 connection->m_stream = TRUE;
a324a7bc
GL
477 connection->m_oriented = TRUE;
478
9bf10d6b 479 /* Setup the peer address field */
85806dc2
GRG
480 connection->m_peer = GAddress_new();
481 if (!connection->m_peer)
482 {
483 GSocket_destroy(connection);
484 socket->m_error = GSOCK_MEMERR;
485 return NULL;
486 }
487 err = _GAddress_translate_from(connection->m_peer, &from, fromlen);
488 if (err != GSOCK_NOERROR)
489 {
490 GAddress_destroy(connection->m_peer);
491 GSocket_destroy(connection);
492 socket->m_error = err;
493 return NULL;
494 }
495
483249fc
GRG
496 ioctl(connection->m_fd, FIONBIO, &arg);
497 _GSocket_Enable_Events(connection);
2f7c2af5 498
a324a7bc
GL
499 return connection;
500}
501
a324a7bc
GL
502/* Client specific parts */
503
2f7c2af5 504/* GSocket_Connect:
9bf10d6b
GRG
505 * For stream (connection oriented) sockets, GSocket_Connect() tries
506 * to establish a client connection to a server using the peer address
507 * as established with GSocket_SetPeer(). Returns GSOCK_NOERROR if the
508 * connection has been succesfully established, or one of the error
509 * codes listed below. Note that for nonblocking sockets, a return
510 * value of GSOCK_WOULDBLOCK doesn't mean a failure. The connection
511 * request can be completed later; you should use GSocket_Select()
512 * to poll for GSOCK_CONNECTION | GSOCK_LOST, or wait for the
513 * corresponding asynchronous events.
514 *
515 * For datagram (non connection oriented) sockets, GSocket_Connect()
516 * just sets the peer address established with GSocket_SetPeer() as
517 * default destination.
518 *
519 * Error codes:
520 * GSOCK_INVSOCK - the socket is in use or not valid.
521 * GSOCK_INVADDR - the peer address has not been established.
522 * GSOCK_TIMEDOUT - timeout, the connection failed.
523 * GSOCK_WOULDBLOCK - connection in progress (nonblocking sockets only)
524 * GSOCK_MEMERR - couldn't allocate memory.
525 * GSOCK_IOERR - low-level error.
2f7c2af5 526 */
a324a7bc
GL
527GSocketError GSocket_Connect(GSocket *sck, GSocketStream stream)
528{
9bf10d6b 529 int err, ret;
483249fc 530 int arg = 1;
a324a7bc
GL
531
532 assert(sck != NULL);
533
483249fc
GRG
534 /* Enable CONNECTION events (needed for nonblocking connections) */
535 _GSocket_Enable(sck, GSOCK_CONNECTION);
536
aac67d4c 537 if (sck->m_fd != INVALID_SOCKET)
483249fc 538 {
a324a7bc
GL
539 sck->m_error = GSOCK_INVSOCK;
540 return GSOCK_INVSOCK;
541 }
542
483249fc
GRG
543 if (!sck->m_peer)
544 {
a324a7bc
GL
545 sck->m_error = GSOCK_INVADDR;
546 return GSOCK_INVADDR;
547 }
548
9bf10d6b 549 /* Streamed or dgram socket? */
483249fc 550 sck->m_stream = (stream == GSOCK_STREAMED);
a324a7bc 551 sck->m_oriented = TRUE;
483249fc
GRG
552 sck->m_server = FALSE;
553 sck->m_establishing = FALSE;
a324a7bc 554
f61815af 555 /* Create the socket */
9bf10d6b
GRG
556 sck->m_fd = socket(sck->m_peer->m_realfamily,
557 sck->m_stream? SOCK_STREAM : SOCK_DGRAM, 0);
a324a7bc 558
aac67d4c 559 if (sck->m_fd == INVALID_SOCKET)
9bf10d6b 560 {
a324a7bc
GL
561 sck->m_error = GSOCK_IOERR;
562 return GSOCK_IOERR;
563 }
564
483249fc
GRG
565 ioctl(sck->m_fd, FIONBIO, &arg);
566 _GSocket_Enable_Events(sck);
aa6d9706 567
9bf10d6b 568 /* Connect it to the peer address, with a timeout (see below) */
483249fc
GRG
569 ret = connect(sck->m_fd, sck->m_peer->m_addr, sck->m_peer->m_len);
570
571 if (ret == -1)
572 {
573 err = errno;
574
575 /* If connect failed with EINPROGRESS and the GSocket object
576 * is in blocking mode, we select() for the specified timeout
577 * checking for writability to see if the connection request
578 * completes.
579 */
580 if ((err == EINPROGRESS) && (!sck->m_non_blocking))
581 {
582 if (_GSocket_Output_Timeout(sck) == GSOCK_TIMEDOUT)
583 {
007c77ab 584 GSocket_close(sck);
483249fc 585 /* sck->m_error is set in _GSocket_Output_Timeout */
483249fc
GRG
586 return GSOCK_TIMEDOUT;
587 }
588 else
589 {
9bf10d6b
GRG
590 int error;
591 SOCKLEN_T len = sizeof(error);
592
593 getsockopt(sck->m_fd, SOL_SOCKET, SO_ERROR, (void*) &error, &len);
594
595 if (!error)
596 return GSOCK_NOERROR;
483249fc
GRG
597 }
598 }
599
600 /* If connect failed with EINPROGRESS and the GSocket object
601 * is set to nonblocking, we set m_error to GSOCK_WOULDBLOCK
602 * (and return GSOCK_WOULDBLOCK) but we don't close the socket;
603 * this way if the connection completes, a GSOCK_CONNECTION
604 * event will be generated, if enabled.
605 */
606 if ((err == EINPROGRESS) && (sck->m_non_blocking))
607 {
483249fc 608 sck->m_establishing = TRUE;
9bf10d6b 609 sck->m_error = GSOCK_WOULDBLOCK;
483249fc
GRG
610 return GSOCK_WOULDBLOCK;
611 }
aa6d9706 612
483249fc
GRG
613 /* If connect failed with an error other than EINPROGRESS,
614 * then the call to GSocket_Connect has failed.
615 */
007c77ab 616 GSocket_close(sck);
aac67d4c
GRG
617 sck->m_error = GSOCK_IOERR;
618 return GSOCK_IOERR;
619 }
620
621 return GSOCK_NOERROR;
622}
623
624/* Datagram sockets */
625
626/* GSocket_SetNonOriented:
627 * Sets up this socket as a non-connection oriented (datagram) socket.
628 * Before using this function, the local address must have been set
629 * with GSocket_SetLocal(), or the call will fail. Returns GSOCK_NOERROR
630 * on success, or one of the following otherwise.
631 *
632 * Error codes:
633 * GSOCK_INVSOCK - the socket is in use.
634 * GSOCK_INVADDR - the local address has not been set.
635 * GSOCK_IOERR - low-level error.
636 */
637GSocketError GSocket_SetNonOriented(GSocket *sck)
638{
639 int arg = 1;
640
641 assert(sck != NULL);
642
643 if (sck->m_fd != INVALID_SOCKET)
644 {
645 sck->m_error = GSOCK_INVSOCK;
646 return GSOCK_INVSOCK;
647 }
648
649 if (!sck->m_local)
650 {
651 sck->m_error = GSOCK_INVADDR;
652 return GSOCK_INVADDR;
653 }
654
655 /* Initialize all fields */
656 sck->m_stream = FALSE;
657 sck->m_server = FALSE;
658 sck->m_oriented = FALSE;
659
660 /* Create the socket */
661 sck->m_fd = socket(sck->m_local->m_realfamily, SOCK_DGRAM, 0);
662
663 if (sck->m_fd == INVALID_SOCKET)
664 {
665 sck->m_error = GSOCK_IOERR;
666 return GSOCK_IOERR;
667 }
668
669 ioctl(sck->m_fd, FIONBIO, &arg);
670 _GSocket_Enable_Events(sck);
671
672 /* Bind to the local address,
673 * and retrieve the actual address bound.
674 */
675 if ((bind(sck->m_fd, sck->m_local->m_addr, sck->m_local->m_len) != 0) ||
676 (getsockname(sck->m_fd,
677 sck->m_local->m_addr,
678 (SOCKLEN_T *) &sck->m_local->m_len) != 0))
679 {
007c77ab 680 GSocket_close(sck);
483249fc 681 sck->m_error = GSOCK_IOERR;
a324a7bc
GL
682 return GSOCK_IOERR;
683 }
39b91eca 684
483249fc 685 return GSOCK_NOERROR;
a324a7bc
GL
686}
687
688/* Generic IO */
689
690/* Like recv(), send(), ... */
691int GSocket_Read(GSocket *socket, char *buffer, int size)
692{
483249fc
GRG
693 int ret;
694
a324a7bc
GL
695 assert(socket != NULL);
696
483249fc
GRG
697 /* Reenable INPUT events */
698 _GSocket_Enable(socket, GSOCK_INPUT);
699
aac67d4c 700 if (socket->m_fd == INVALID_SOCKET || socket->m_server)
483249fc 701 {
a324a7bc
GL
702 socket->m_error = GSOCK_INVSOCK;
703 return -1;
704 }
705
9bf10d6b 706 /* If the socket is blocking, wait for data (with a timeout) */
483249fc
GRG
707 if (_GSocket_Input_Timeout(socket) == GSOCK_TIMEDOUT)
708 return -1;
a324a7bc 709
9bf10d6b 710 /* Read the data */
f61815af 711 if (socket->m_stream)
483249fc 712 ret = _GSocket_Recv_Stream(socket, buffer, size);
a324a7bc 713 else
483249fc
GRG
714 ret = _GSocket_Recv_Dgram(socket, buffer, size);
715
716 if (ret == -1)
717 {
718 if (errno == EWOULDBLOCK)
719 socket->m_error = GSOCK_WOULDBLOCK;
720 else
721 socket->m_error = GSOCK_IOERR;
722 }
723
724 return ret;
a324a7bc
GL
725}
726
9bf10d6b
GRG
727int GSocket_Write(GSocket *socket, const char *buffer, int size)
728{
483249fc
GRG
729 int ret;
730
a324a7bc 731 assert(socket != NULL);
856bc99a 732
8e907a13 733 GSocket_Debug(( "GSocket_Write #1, size %d\n", size ));
a324a7bc 734
aac67d4c 735 if (socket->m_fd == INVALID_SOCKET || socket->m_server)
483249fc 736 {
a324a7bc
GL
737 socket->m_error = GSOCK_INVSOCK;
738 return -1;
739 }
740
8e907a13 741 GSocket_Debug(( "GSocket_Write #2, size %d\n", size ));
856bc99a 742
9bf10d6b 743 /* If the socket is blocking, wait for writability (with a timeout) */
483249fc
GRG
744 if (_GSocket_Output_Timeout(socket) == GSOCK_TIMEDOUT)
745 return -1;
a324a7bc 746
8e907a13 747 GSocket_Debug(( "GSocket_Write #3, size %d\n", size ));
856bc99a 748
9bf10d6b 749 /* Write the data */
f61815af 750 if (socket->m_stream)
483249fc 751 ret = _GSocket_Send_Stream(socket, buffer, size);
a324a7bc 752 else
483249fc
GRG
753 ret = _GSocket_Send_Dgram(socket, buffer, size);
754
8e907a13 755 GSocket_Debug(( "GSocket_Write #4, size %d\n", size ));
856bc99a 756
483249fc
GRG
757 if (ret == -1)
758 {
759 if (errno == EWOULDBLOCK)
d5ab387d 760 {
483249fc 761 socket->m_error = GSOCK_WOULDBLOCK;
8e907a13 762 GSocket_Debug(( "GSocket_Write error WOULDBLOCK\n" ));
d5ab387d 763 }
483249fc 764 else
d5ab387d 765 {
483249fc 766 socket->m_error = GSOCK_IOERR;
8e907a13 767 GSocket_Debug(( "GSocket_Write error IOERR\n" ));
d5ab387d 768 }
483249fc
GRG
769
770 /* Only reenable OUTPUT events after an error (just like WSAAsyncSelect
771 * in MSW). Once the first OUTPUT event is received, users can assume
772 * that the socket is writable until a read operation fails. Only then
773 * will further OUTPUT events be posted.
774 */
775 _GSocket_Enable(socket, GSOCK_OUTPUT);
9bf10d6b 776 return -1;
483249fc
GRG
777 }
778
8e907a13 779 GSocket_Debug(( "GSocket_Write #5, size %d ret %d\n", size, ret ));
856bc99a 780
483249fc 781 return ret;
a324a7bc
GL
782}
783
ef25e638
GRG
784/* GSocket_Select:
785 * Polls the socket to determine its status. This function will
786 * check for the events specified in the 'flags' parameter, and
787 * it will return a mask indicating which operations can be
788 * performed. This function won't block, regardless of the
9bf10d6b 789 * mode (blocking | nonblocking) of the socket.
ef25e638
GRG
790 */
791GSocketEventFlags GSocket_Select(GSocket *socket, GSocketEventFlags flags)
a324a7bc 792{
8e907a13
VZ
793#if defined(wxUSE_GUI) && !wxUSE_GUI
794
795 GSocketEventFlags result = 0;
796 fd_set readfds;
797 fd_set writefds;
798 fd_set exceptfds;
799 struct timeval tv;
800
801 /* Do not use a static struct, Linux can garble it */
802 tv.tv_sec = 0;
803 tv.tv_usec = 0;
804
a324a7bc
GL
805 assert(socket != NULL);
806
8e907a13
VZ
807 FD_ZERO(&readfds);
808 FD_ZERO(&writefds);
809 FD_ZERO(&exceptfds);
810 FD_SET(socket->m_fd, &readfds);
811 FD_SET(socket->m_fd, &writefds);
812 FD_SET(socket->m_fd, &exceptfds);
813
814 /* Check known state first */
815 result |= (GSOCK_CONNECTION_FLAG & socket->m_detected & flags);
816 result |= (GSOCK_LOST_FLAG & socket->m_detected & flags);
817
818 /* Try select now */
819 if (select(socket->m_fd + 1, &readfds, &writefds, &exceptfds, &tv) <= 0)
820 return result;
821
822 /* Check for readability */
823 if (FD_ISSET(socket->m_fd, &readfds))
824 {
825 char c;
826
827 if (recv(socket->m_fd, &c, 1, MSG_PEEK) > 0)
828 {
829 result |= (GSOCK_INPUT_FLAG & flags);
830 }
831 else
832 {
833 if (socket->m_server && socket->m_stream)
834 {
835 result |= (GSOCK_CONNECTION_FLAG & flags);
836 socket->m_detected |= GSOCK_CONNECTION_FLAG;
837 }
838 else
839 {
840 result |= (GSOCK_LOST_FLAG & flags);
841 socket->m_detected = GSOCK_LOST_FLAG;
842 }
843 }
844 }
845
846 /* Check for writability */
847 if (FD_ISSET(socket->m_fd, &writefds))
848 {
849 if (socket->m_establishing && !socket->m_server)
850 {
851 int error;
852 SOCKLEN_T len = sizeof(error);
853
854 socket->m_establishing = FALSE;
855
856 getsockopt(socket->m_fd, SOL_SOCKET, SO_ERROR, (void*)&error, &len);
857
858 if (error)
859 {
860 result |= (GSOCK_LOST_FLAG & flags);
861 socket->m_detected = GSOCK_LOST_FLAG;
862 }
863 else
864 {
865 result |= (GSOCK_CONNECTION_FLAG & flags);
866 socket->m_detected |= GSOCK_CONNECTION_FLAG;
867 }
868 }
869 else
870 {
871 result |= (GSOCK_OUTPUT_FLAG & flags);
872 }
873 }
874
875 /* Check for exceptions and errors (is this useful in Unices?) */
876 if (FD_ISSET(socket->m_fd, &exceptfds))
877 {
878 result |= (GSOCK_LOST_FLAG & flags);
879 socket->m_establishing = FALSE;
880 socket->m_detected = GSOCK_LOST_FLAG;
881 }
882
883 return result;
884
885#else
886
887 assert(socket != NULL);
ef57d866 888 return flags & socket->m_detected;
8e907a13
VZ
889
890#endif /* !wxUSE_GUI */
a324a7bc
GL
891}
892
893/* Flags */
894
2f7c2af5 895/* GSocket_SetNonBlocking:
9bf10d6b
GRG
896 * Sets the socket to non-blocking mode. All IO calls will return
897 * immediately.
2f7c2af5 898 */
5c9eff30 899void GSocket_SetNonBlocking(GSocket *socket, int non_block)
a324a7bc
GL
900{
901 assert(socket != NULL);
902
609c60af
GRG
903 GSocket_Debug( ("GSocket_SetNonBlocking: %d\n", (int)non_block) );
904
dc26b37a 905 socket->m_non_blocking = non_block;
a324a7bc
GL
906}
907
2f7c2af5 908/* GSocket_SetTimeout:
9bf10d6b
GRG
909 * Sets the timeout for blocking calls. Time is expressed in
910 * milliseconds.
39b91eca
GL
911 */
912void GSocket_SetTimeout(GSocket *socket, unsigned long millisec)
913{
914 assert(socket != NULL);
915
54e575f9 916 socket->m_timeout = millisec;
39b91eca
GL
917}
918
2f7c2af5 919/* GSocket_GetError:
9bf10d6b
GRG
920 * Returns the last error occured for this socket. Note that successful
921 * operations do not clear this back to GSOCK_NOERROR, so use it only
922 * after an error.
2f7c2af5 923 */
a324a7bc
GL
924GSocketError GSocket_GetError(GSocket *socket)
925{
926 assert(socket != NULL);
927
928 return socket->m_error;
929}
930
931/* Callbacks */
932
9bf10d6b
GRG
933/* GSOCK_INPUT:
934 * There is data to be read in the input buffer. If, after a read
935 * operation, there is still data available, the callback function will
936 * be called again.
937 * GSOCK_OUTPUT:
938 * The socket is available for writing. That is, the next write call
939 * won't block. This event is generated only once, when the connection is
940 * first established, and then only if a call failed with GSOCK_WOULDBLOCK,
941 * when the output buffer empties again. This means that the app should
942 * assume that it can write since the first OUTPUT event, and no more
943 * OUTPUT events will be generated unless an error occurs.
944 * GSOCK_CONNECTION:
945 * Connection succesfully established, for client sockets, or incoming
946 * client connection, for server sockets. Wait for this event (also watch
947 * out for GSOCK_LOST) after you issue a nonblocking GSocket_Connect() call.
948 * GSOCK_LOST:
949 * The connection is lost (or a connection request failed); this could
950 * be due to a failure, or due to the peer closing it gracefully.
2f7c2af5
GRG
951 */
952
953/* GSocket_SetCallback:
954 * Enables the callbacks specified by 'flags'. Note that 'flags'
955 * may be a combination of flags OR'ed toghether, so the same
956 * callback function can be made to accept different events.
957 * The callback function must have the following prototype:
958 *
959 * void function(GSocket *socket, GSocketEvent event, char *cdata)
960 */
ef25e638 961void GSocket_SetCallback(GSocket *socket, GSocketEventFlags flags,
9bf10d6b 962 GSocketCallback callback, char *cdata)
a324a7bc
GL
963{
964 int count;
965
483249fc 966 assert(socket != NULL);
a324a7bc 967
483249fc
GRG
968 for (count = 0; count < GSOCK_MAX_EVENT; count++)
969 {
970 if ((flags & (1 << count)) != 0)
971 {
98781fa3 972 socket->m_cbacks[count] = callback;
a324a7bc 973 socket->m_data[count] = cdata;
a324a7bc
GL
974 }
975 }
976}
977
2f7c2af5
GRG
978/* GSocket_UnsetCallback:
979 * Disables all callbacks specified by 'flags', which may be a
980 * combination of flags OR'ed toghether.
981 */
ef25e638 982void GSocket_UnsetCallback(GSocket *socket, GSocketEventFlags flags)
a324a7bc 983{
2f7c2af5 984 int count;
a324a7bc
GL
985
986 assert(socket != NULL);
987
483249fc
GRG
988 for (count = 0; count < GSOCK_MAX_EVENT; count++)
989 {
990 if ((flags & (1 << count)) != 0)
991 {
98781fa3 992 socket->m_cbacks[count] = NULL;
483249fc 993 socket->m_data[count] = NULL;
a324a7bc
GL
994 }
995 }
996}
997
9bf10d6b 998
483249fc
GRG
999#define CALL_CALLBACK(socket, event) { \
1000 _GSocket_Disable(socket, event); \
1001 if (socket->m_cbacks[event]) \
1002 socket->m_cbacks[event](socket, event, socket->m_data[event]); \
a324a7bc
GL
1003}
1004
483249fc 1005
a324a7bc
GL
1006void _GSocket_Enable(GSocket *socket, GSocketEvent event)
1007{
483249fc
GRG
1008 socket->m_detected &= ~(1 << event);
1009 _GSocket_Install_Callback(socket, event);
a324a7bc
GL
1010}
1011
1012void _GSocket_Disable(GSocket *socket, GSocketEvent event)
1013{
483249fc
GRG
1014 socket->m_detected |= (1 << event);
1015 _GSocket_Uninstall_Callback(socket, event);
a324a7bc
GL
1016}
1017
483249fc
GRG
1018/* _GSocket_Input_Timeout:
1019 * For blocking sockets, wait until data is available or
1020 * until timeout ellapses.
1021 */
1022GSocketError _GSocket_Input_Timeout(GSocket *socket)
2f7c2af5 1023{
483249fc
GRG
1024 struct timeval tv;
1025 fd_set readfds;
a8a0b892 1026 int ret;
483249fc
GRG
1027
1028 /* Linux select() will overwrite the struct on return */
1029 tv.tv_sec = (socket->m_timeout / 1000);
1030 tv.tv_usec = (socket->m_timeout % 1000) * 1000;
1031
1032 if (!socket->m_non_blocking)
1033 {
1034 FD_ZERO(&readfds);
1035 FD_SET(socket->m_fd, &readfds);
a8a0b892
RR
1036 ret = select(socket->m_fd + 1, &readfds, NULL, NULL, &tv);
1037 if (ret == 0)
1038 {
8e907a13 1039 GSocket_Debug(( "GSocket_Input_Timeout, select returned 0\n" ));
a8a0b892
RR
1040 socket->m_error = GSOCK_TIMEDOUT;
1041 return GSOCK_TIMEDOUT;
1042 }
1043 if (ret == -1)
483249fc 1044 {
8e907a13 1045 GSocket_Debug(( "GSocket_Input_Timeout, select returned -1\n" ));
d84afea9
GD
1046 if (errno == EBADF) { GSocket_Debug(( "Invalid file descriptor\n" )); }
1047 if (errno == EINTR) { GSocket_Debug(( "A non blocked signal was caught\n" )); }
1048 if (errno == EINVAL) { GSocket_Debug(( "The highest number descriptor is negative\n" )); }
1049 if (errno == ENOMEM) { GSocket_Debug(( "Not enough memory\n" )); }
483249fc
GRG
1050 socket->m_error = GSOCK_TIMEDOUT;
1051 return GSOCK_TIMEDOUT;
1052 }
1053 }
1054 return GSOCK_NOERROR;
1055}
1056
1057/* _GSocket_Output_Timeout:
1058 * For blocking sockets, wait until data can be sent without
1059 * blocking or until timeout ellapses.
1060 */
1061GSocketError _GSocket_Output_Timeout(GSocket *socket)
1062{
1063 struct timeval tv;
1064 fd_set writefds;
a8a0b892 1065 int ret;
483249fc
GRG
1066
1067 /* Linux select() will overwrite the struct on return */
1068 tv.tv_sec = (socket->m_timeout / 1000);
1069 tv.tv_usec = (socket->m_timeout % 1000) * 1000;
1070
609c60af
GRG
1071 GSocket_Debug( ("m_non_blocking has: %d\n", (int)socket->m_non_blocking) );
1072
483249fc
GRG
1073 if (!socket->m_non_blocking)
1074 {
1075 FD_ZERO(&writefds);
1076 FD_SET(socket->m_fd, &writefds);
a8a0b892
RR
1077 ret = select(socket->m_fd + 1, NULL, &writefds, NULL, &tv);
1078 if (ret == 0)
1079 {
8e907a13 1080 GSocket_Debug(( "GSocket_Output_Timeout, select returned 0\n" ));
a8a0b892
RR
1081 socket->m_error = GSOCK_TIMEDOUT;
1082 return GSOCK_TIMEDOUT;
1083 }
1084 if (ret == -1)
483249fc 1085 {
8e907a13 1086 GSocket_Debug(( "GSocket_Output_Timeout, select returned -1\n" ));
d84afea9
GD
1087 if (errno == EBADF) { GSocket_Debug(( "Invalid file descriptor\n" )); }
1088 if (errno == EINTR) { GSocket_Debug(( "A non blocked signal was caught\n" )); }
1089 if (errno == EINVAL) { GSocket_Debug(( "The highest number descriptor is negative\n" )); }
1090 if (errno == ENOMEM) { GSocket_Debug(( "Not enough memory\n" )); }
483249fc
GRG
1091 socket->m_error = GSOCK_TIMEDOUT;
1092 return GSOCK_TIMEDOUT;
2f7c2af5 1093 }
d84afea9
GD
1094 if ( ! FD_ISSET(socket->m_fd, &writefds) ) {
1095 GSocket_Debug(( "GSocket_Output_Timeout is buggy!\n" ));
1096 }
1097 else {
1098 GSocket_Debug(( "GSocket_Output_Timeout seems correct\n" ));
1099 }
2f7c2af5 1100 }
aac67d4c
GRG
1101 else
1102 {
8e3aec85 1103 GSocket_Debug(( "GSocket_Output_Timeout, didn't try select!\n" ));
aac67d4c
GRG
1104 }
1105
483249fc 1106 return GSOCK_NOERROR;
2f7c2af5
GRG
1107}
1108
a324a7bc
GL
1109int _GSocket_Recv_Stream(GSocket *socket, char *buffer, int size)
1110{
04e1eb03 1111 return recv(socket->m_fd, buffer, size, 0);
a324a7bc
GL
1112}
1113
1114int _GSocket_Recv_Dgram(GSocket *socket, char *buffer, int size)
1115{
1116 struct sockaddr from;
85806dc2 1117 SOCKLEN_T fromlen = sizeof(from);
ca17eff3 1118 int ret;
aa6d9706 1119 GSocketError err;
a324a7bc
GL
1120
1121 fromlen = sizeof(from);
1122
85806dc2 1123 ret = recvfrom(socket->m_fd, buffer, size, 0, &from, (SOCKLEN_T *) &fromlen);
483249fc
GRG
1124
1125 if (ret == -1)
aa6d9706 1126 return -1;
a324a7bc 1127
f61815af 1128 /* Translate a system address into a GSocket address */
483249fc
GRG
1129 if (!socket->m_peer)
1130 {
a324a7bc 1131 socket->m_peer = GAddress_new();
483249fc
GRG
1132 if (!socket->m_peer)
1133 {
e00f35bb
GL
1134 socket->m_error = GSOCK_MEMERR;
1135 return -1;
1136 }
1137 }
aa6d9706 1138 err = _GAddress_translate_from(socket->m_peer, &from, fromlen);
483249fc
GRG
1139 if (err != GSOCK_NOERROR)
1140 {
aa6d9706
GL
1141 GAddress_destroy(socket->m_peer);
1142 socket->m_peer = NULL;
1143 socket->m_error = err;
e00f35bb 1144 return -1;
aa6d9706 1145 }
a324a7bc
GL
1146
1147 return ret;
1148}
1149
1150int _GSocket_Send_Stream(GSocket *socket, const char *buffer, int size)
1151{
1152 int ret;
1153
1154 MASK_SIGNAL();
1155 ret = send(socket->m_fd, buffer, size, 0);
1156 UNMASK_SIGNAL();
483249fc 1157
a324a7bc
GL
1158 return ret;
1159}
1160
1161int _GSocket_Send_Dgram(GSocket *socket, const char *buffer, int size)
1162{
1163 struct sockaddr *addr;
1164 int len, ret;
aa6d9706 1165 GSocketError err;
a324a7bc 1166
9bf10d6b
GRG
1167 if (!socket->m_peer)
1168 {
a324a7bc
GL
1169 socket->m_error = GSOCK_INVADDR;
1170 return -1;
1171 }
1172
aa6d9706 1173 err = _GAddress_translate_to(socket->m_peer, &addr, &len);
9bf10d6b
GRG
1174 if (err != GSOCK_NOERROR)
1175 {
aa6d9706 1176 socket->m_error = err;
e00f35bb
GL
1177 return -1;
1178 }
a324a7bc
GL
1179
1180 MASK_SIGNAL();
1181 ret = sendto(socket->m_fd, buffer, size, 0, addr, len);
1182 UNMASK_SIGNAL();
a324a7bc 1183
f61815af 1184 /* Frees memory allocated from _GAddress_translate_to */
a324a7bc
GL
1185 free(addr);
1186
1187 return ret;
1188}
1189
1190void _GSocket_Detected_Read(GSocket *socket)
1191{
1192 char c;
9bf10d6b 1193
8e907a13 1194 if (recv(socket->m_fd, &c, 1, MSG_PEEK) > 0)
9bf10d6b
GRG
1195 {
1196 CALL_CALLBACK(socket, GSOCK_INPUT);
1197 }
1198 else
1199 {
8e907a13
VZ
1200 if (socket->m_server && socket->m_stream)
1201 {
1202 CALL_CALLBACK(socket, GSOCK_CONNECTION);
1203 }
1204 else
1205 {
1206 CALL_CALLBACK(socket, GSOCK_LOST);
1207 }
a324a7bc
GL
1208 }
1209}
1210
1211void _GSocket_Detected_Write(GSocket *socket)
1212{
483249fc
GRG
1213 if (socket->m_establishing && !socket->m_server)
1214 {
9bf10d6b
GRG
1215 int error;
1216 SOCKLEN_T len = sizeof(error);
aa6d9706 1217
aa6d9706 1218 socket->m_establishing = FALSE;
aa6d9706 1219
8e907a13 1220 getsockopt(socket->m_fd, SOL_SOCKET, SO_ERROR, (void*)&error, &len);
aa6d9706 1221
483249fc
GRG
1222 if (error)
1223 {
1224 CALL_CALLBACK(socket, GSOCK_LOST);
1225 }
1226 else
1227 {
1228 CALL_CALLBACK(socket, GSOCK_CONNECTION);
1229 /* We have to fire this event by hand because CONNECTION (for clients)
1230 * and OUTPUT are internally the same and we just disabled CONNECTION
1231 * events with the above macro.
1232 */
1233 CALL_CALLBACK(socket, GSOCK_OUTPUT);
1234 }
1235 }
1236 else
1237 {
aa6d9706 1238 CALL_CALLBACK(socket, GSOCK_OUTPUT);
483249fc 1239 }
aa6d9706 1240}
a324a7bc
GL
1241
1242/*
1243 * -------------------------------------------------------------------------
1244 * GAddress
1245 * -------------------------------------------------------------------------
1246 */
1247
032d5581
GRG
1248/* CHECK_ADDRESS verifies that the current address family is either
1249 * GSOCK_NOFAMILY or GSOCK_*family*, and if it is GSOCK_NOFAMILY, it
1250 * initalizes it to be a GSOCK_*family*. In other cases, it returns
1251 * an appropiate error code.
1252 *
1253 * CHECK_ADDRESS_RETVAL does the same but returning 'retval' on error.
f61815af 1254 */
032d5581 1255#define CHECK_ADDRESS(address, family) \
9bf10d6b
GRG
1256{ \
1257 if (address->m_family == GSOCK_NOFAMILY) \
1258 if (_GAddress_Init_##family(address) != GSOCK_NOERROR) \
1259 return address->m_error; \
1260 if (address->m_family != GSOCK_##family) \
032d5581
GRG
1261 { \
1262 address->m_error = GSOCK_INVADDR; \
1263 return GSOCK_INVADDR; \
1264 } \
1265}
1266
1267#define CHECK_ADDRESS_RETVAL(address, family, retval) \
1268{ \
1269 if (address->m_family == GSOCK_NOFAMILY) \
1270 if (_GAddress_Init_##family(address) != GSOCK_NOERROR) \
1271 return retval; \
1272 if (address->m_family != GSOCK_##family) \
9bf10d6b
GRG
1273 { \
1274 address->m_error = GSOCK_INVADDR; \
1275 return retval; \
1276 } \
a324a7bc
GL
1277}
1278
032d5581 1279
da051b23 1280GAddress *GAddress_new(void)
a324a7bc
GL
1281{
1282 GAddress *address;
1283
9bf10d6b 1284 if ((address = (GAddress *) malloc(sizeof(GAddress))) == NULL)
e00f35bb
GL
1285 return NULL;
1286
a324a7bc
GL
1287 address->m_family = GSOCK_NOFAMILY;
1288 address->m_addr = NULL;
1289 address->m_len = 0;
1290
1291 return address;
1292}
1293
1294GAddress *GAddress_copy(GAddress *address)
1295{
1296 GAddress *addr2;
1297
1298 assert(address != NULL);
1299
9bf10d6b 1300 if ((addr2 = (GAddress *) malloc(sizeof(GAddress))) == NULL)
e00f35bb
GL
1301 return NULL;
1302
a324a7bc
GL
1303 memcpy(addr2, address, sizeof(GAddress));
1304
007c77ab 1305 if (address->m_addr && address->m_len > 0)
9bf10d6b 1306 {
a324a7bc 1307 addr2->m_addr = (struct sockaddr *)malloc(addr2->m_len);
9bf10d6b
GRG
1308 if (addr2->m_addr == NULL)
1309 {
e00f35bb
GL
1310 free(addr2);
1311 return NULL;
1312 }
a324a7bc
GL
1313 memcpy(addr2->m_addr, address->m_addr, addr2->m_len);
1314 }
1315
1316 return addr2;
1317}
1318
1319void GAddress_destroy(GAddress *address)
1320{
1321 assert(address != NULL);
1322
29c25a8e
GRG
1323 if (address->m_addr)
1324 free(address->m_addr);
1325
a324a7bc
GL
1326 free(address);
1327}
1328
1329void GAddress_SetFamily(GAddress *address, GAddressType type)
1330{
1331 assert(address != NULL);
1332
1333 address->m_family = type;
1334}
1335
1336GAddressType GAddress_GetFamily(GAddress *address)
1337{
1338 assert(address != NULL);
1339
1340 return address->m_family;
1341}
1342
9bf10d6b
GRG
1343GSocketError _GAddress_translate_from(GAddress *address,
1344 struct sockaddr *addr, int len)
1345{
a324a7bc 1346 address->m_realfamily = addr->sa_family;
9bf10d6b
GRG
1347 switch (addr->sa_family)
1348 {
1349 case AF_INET:
1350 address->m_family = GSOCK_INET;
1351 break;
1352 case AF_UNIX:
1353 address->m_family = GSOCK_UNIX;
1354 break;
efee48a0 1355#ifdef AF_INET6
9bf10d6b
GRG
1356 case AF_INET6:
1357 address->m_family = GSOCK_INET6;
1358 break;
efee48a0 1359#endif
9bf10d6b 1360 default:
ca17eff3 1361 {
9bf10d6b
GRG
1362 address->m_error = GSOCK_INVOP;
1363 return GSOCK_INVOP;
ca17eff3 1364 }
a324a7bc
GL
1365 }
1366
1367 if (address->m_addr)
1368 free(address->m_addr);
1369
1370 address->m_len = len;
1371 address->m_addr = (struct sockaddr *)malloc(len);
9bf10d6b
GRG
1372
1373 if (address->m_addr == NULL)
1374 {
54e575f9
GL
1375 address->m_error = GSOCK_MEMERR;
1376 return GSOCK_MEMERR;
1377 }
a324a7bc 1378 memcpy(address->m_addr, addr, len);
e00f35bb 1379
54e575f9 1380 return GSOCK_NOERROR;
a324a7bc
GL
1381}
1382
54e575f9
GL
1383GSocketError _GAddress_translate_to(GAddress *address,
1384 struct sockaddr **addr, int *len)
a324a7bc 1385{
9bf10d6b
GRG
1386 if (!address->m_addr)
1387 {
54e575f9
GL
1388 address->m_error = GSOCK_INVADDR;
1389 return GSOCK_INVADDR;
a324a7bc
GL
1390 }
1391
1392 *len = address->m_len;
1393 *addr = (struct sockaddr *)malloc(address->m_len);
032d5581
GRG
1394 if (*addr == NULL)
1395 {
54e575f9
GL
1396 address->m_error = GSOCK_MEMERR;
1397 return GSOCK_MEMERR;
1398 }
e00f35bb 1399
a324a7bc 1400 memcpy(*addr, address->m_addr, address->m_len);
54e575f9 1401 return GSOCK_NOERROR;
a324a7bc
GL
1402}
1403
1404/*
1405 * -------------------------------------------------------------------------
1406 * Internet address family
1407 * -------------------------------------------------------------------------
1408 */
1409
54e575f9 1410GSocketError _GAddress_Init_INET(GAddress *address)
a324a7bc 1411{
f3e60952 1412 address->m_len = sizeof(struct sockaddr_in);
9bf10d6b 1413 address->m_addr = (struct sockaddr *) malloc(address->m_len);
f3e60952
GRG
1414 if (address->m_addr == NULL)
1415 {
54e575f9
GL
1416 address->m_error = GSOCK_MEMERR;
1417 return GSOCK_MEMERR;
1418 }
e00f35bb 1419
a324a7bc
GL
1420 address->m_family = GSOCK_INET;
1421 address->m_realfamily = PF_INET;
1422 ((struct sockaddr_in *)address->m_addr)->sin_family = AF_INET;
9bf10d6b 1423 ((struct sockaddr_in *)address->m_addr)->sin_addr.s_addr = INADDR_ANY;
e00f35bb 1424
cc345f78 1425 return GSOCK_NOERROR;
a324a7bc
GL
1426}
1427
1428GSocketError GAddress_INET_SetHostName(GAddress *address, const char *hostname)
1429{
1430 struct hostent *he;
1431 struct in_addr *addr;
1432
1433 assert(address != NULL);
1434
032d5581 1435 CHECK_ADDRESS(address, INET);
a324a7bc
GL
1436
1437 addr = &(((struct sockaddr_in *)address->m_addr)->sin_addr);
1438
f61815af 1439 /* If it is a numeric host name, convert it now */
d5b08e7b 1440#if defined(HAVE_INET_ATON)
f3e60952
GRG
1441 if (inet_aton(hostname, addr) == 0)
1442 {
d5b08e7b 1443#elif defined(HAVE_INET_ADDR)
f3e60952
GRG
1444 if ( (addr->s_addr = inet_addr(hostname)) == -1 )
1445 {
e96ac54e
GL
1446#else
1447 /* Use gethostbyname by default */
f3e60952
GRG
1448 if (1)
1449 {
e96ac54e 1450#endif
a324a7bc
GL
1451 struct in_addr *array_addr;
1452
f61815af 1453 /* It is a real name, we solve it */
f3e60952
GRG
1454 if ((he = gethostbyname(hostname)) == NULL)
1455 {
9bf10d6b
GRG
1456 /* Reset to invalid address */
1457 addr->s_addr = INADDR_NONE;
a324a7bc
GL
1458 address->m_error = GSOCK_NOHOST;
1459 return GSOCK_NOHOST;
1460 }
1461 array_addr = (struct in_addr *) *(he->h_addr_list);
1462 addr->s_addr = array_addr[0].s_addr;
1463 }
1464 return GSOCK_NOERROR;
1465}
1466
9bf10d6b
GRG
1467GSocketError GAddress_INET_SetAnyAddress(GAddress *address)
1468{
1469 return GAddress_INET_SetHostAddress(address, INADDR_ANY);
1470}
1471
a324a7bc
GL
1472GSocketError GAddress_INET_SetHostAddress(GAddress *address,
1473 unsigned long hostaddr)
1474{
1475 struct in_addr *addr;
1476
1477 assert(address != NULL);
1478
032d5581 1479 CHECK_ADDRESS(address, INET);
a324a7bc
GL
1480
1481 addr = &(((struct sockaddr_in *)address->m_addr)->sin_addr);
1482 addr->s_addr = hostaddr;
1483
1484 return GSOCK_NOERROR;
1485}
1486
5a96d2f4
GL
1487GSocketError GAddress_INET_SetPortName(GAddress *address, const char *port,
1488 const char *protocol)
a324a7bc
GL
1489{
1490 struct servent *se;
1491 struct sockaddr_in *addr;
1492
1493 assert(address != NULL);
032d5581 1494 CHECK_ADDRESS(address, INET);
a324a7bc 1495
f3e60952
GRG
1496 if (!port)
1497 {
a324a7bc 1498 address->m_error = GSOCK_INVPORT;
54e575f9 1499 return GSOCK_INVPORT;
a324a7bc
GL
1500 }
1501
5a96d2f4 1502 se = getservbyname(port, protocol);
9bf10d6b
GRG
1503 if (!se)
1504 {
72c71d39
VZ
1505 /* the cast to int suppresses compiler warnings about subscript having the
1506 type char */
1507 if (isdigit((int)port[0]))
f3e60952 1508 {
a324a7bc
GL
1509 int port_int;
1510
1511 port_int = atoi(port);
1512 addr = (struct sockaddr_in *)address->m_addr;
1513 addr->sin_port = htons(port_int);
1514 return GSOCK_NOERROR;
1515 }
1516
1517 address->m_error = GSOCK_INVPORT;
1518 return GSOCK_INVPORT;
1519 }
1520
1521 addr = (struct sockaddr_in *)address->m_addr;
1522 addr->sin_port = se->s_port;
1523
1524 return GSOCK_NOERROR;
1525}
1526
1527GSocketError GAddress_INET_SetPort(GAddress *address, unsigned short port)
1528{
1529 struct sockaddr_in *addr;
1530
1531 assert(address != NULL);
032d5581 1532 CHECK_ADDRESS(address, INET);
a324a7bc
GL
1533
1534 addr = (struct sockaddr_in *)address->m_addr;
1535 addr->sin_port = htons(port);
1536
1537 return GSOCK_NOERROR;
1538}
1539
1540GSocketError GAddress_INET_GetHostName(GAddress *address, char *hostname, size_t sbuf)
1541{
1542 struct hostent *he;
1543 char *addr_buf;
1544 struct sockaddr_in *addr;
1545
1546 assert(address != NULL);
032d5581 1547 CHECK_ADDRESS(address, INET);
a324a7bc
GL
1548
1549 addr = (struct sockaddr_in *)address->m_addr;
1550 addr_buf = (char *)&(addr->sin_addr);
1551
f3e60952
GRG
1552 he = gethostbyaddr(addr_buf, sizeof(addr->sin_addr), AF_INET);
1553 if (he == NULL)
1554 {
a324a7bc
GL
1555 address->m_error = GSOCK_NOHOST;
1556 return GSOCK_NOHOST;
1557 }
1558
1559 strncpy(hostname, he->h_name, sbuf);
1560
1561 return GSOCK_NOERROR;
1562}
1563
1564unsigned long GAddress_INET_GetHostAddress(GAddress *address)
1565{
1566 struct sockaddr_in *addr;
1567
1568 assert(address != NULL);
032d5581 1569 CHECK_ADDRESS_RETVAL(address, INET, 0);
a324a7bc
GL
1570
1571 addr = (struct sockaddr_in *)address->m_addr;
1572
1573 return addr->sin_addr.s_addr;
1574}
1575
1576unsigned short GAddress_INET_GetPort(GAddress *address)
1577{
1578 struct sockaddr_in *addr;
1579
1580 assert(address != NULL);
032d5581 1581 CHECK_ADDRESS_RETVAL(address, INET, 0);
a324a7bc
GL
1582
1583 addr = (struct sockaddr_in *)address->m_addr;
1584 return ntohs(addr->sin_port);
1585}
1586
1587/*
1588 * -------------------------------------------------------------------------
1589 * Unix address family
1590 * -------------------------------------------------------------------------
1591 */
1592
54e575f9 1593GSocketError _GAddress_Init_UNIX(GAddress *address)
a324a7bc 1594{
f3e60952 1595 address->m_len = sizeof(struct sockaddr_un);
a324a7bc 1596 address->m_addr = (struct sockaddr *)malloc(address->m_len);
f3e60952
GRG
1597 if (address->m_addr == NULL)
1598 {
54e575f9
GL
1599 address->m_error = GSOCK_MEMERR;
1600 return GSOCK_MEMERR;
1601 }
e00f35bb 1602
a324a7bc
GL
1603 address->m_family = GSOCK_UNIX;
1604 address->m_realfamily = PF_UNIX;
1605 ((struct sockaddr_un *)address->m_addr)->sun_family = AF_UNIX;
1606 ((struct sockaddr_un *)address->m_addr)->sun_path[0] = 0;
e00f35bb 1607
032d5581 1608 return GSOCK_NOERROR;
a324a7bc
GL
1609}
1610
70914290
VZ
1611#define UNIX_SOCK_PATHLEN (sizeof(addr->sun_path)/sizeof(addr->sun_path[0]))
1612
a324a7bc
GL
1613GSocketError GAddress_UNIX_SetPath(GAddress *address, const char *path)
1614{
1615 struct sockaddr_un *addr;
1616
1617 assert(address != NULL);
1618
032d5581 1619 CHECK_ADDRESS(address, UNIX);
a324a7bc
GL
1620
1621 addr = ((struct sockaddr_un *)address->m_addr);
70914290
VZ
1622 strncpy(addr->sun_path, path, UNIX_SOCK_PATHLEN);
1623 addr->sun_path[UNIX_SOCK_PATHLEN - 1] = '\0';
a324a7bc
GL
1624
1625 return GSOCK_NOERROR;
1626}
1627
1628GSocketError GAddress_UNIX_GetPath(GAddress *address, char *path, size_t sbuf)
1629{
1630 struct sockaddr_un *addr;
1631
1632 assert(address != NULL);
032d5581 1633 CHECK_ADDRESS(address, UNIX);
a324a7bc
GL
1634
1635 addr = (struct sockaddr_un *)address->m_addr;
1636
1637 strncpy(path, addr->sun_path, sbuf);
1638
1639 return GSOCK_NOERROR;
1640}
813c20a6 1641
9bf10d6b 1642#endif /* wxUSE_SOCKETS || defined(__GSOCKET_STANDALONE__) */
85806dc2 1643
99f72806
MB
1644/* vi:sts=4:sw=4:et */
1645