]> git.saurik.com Git - wxWidgets.git/blob - src/mac/carbon/gsocket.cpp
mingw evidently doesn't have digitalv.h according to mmedia sources - post a workaround
[wxWidgets.git] / src / mac / carbon / gsocket.cpp
1 /* -------------------------------------------------------------------------
2 * Project: GSocket (Generic Socket) for WX
3 * Name: gsocket.cpp
4 * Authors: Guilhem Lavaux,
5 * Guillermo Rodriguez Garcia <guille@iies.es> (maintainer)
6 * Stefan CSomor
7 * Purpose: GSocket main mac file
8 * CVSID: $Id$
9 * -------------------------------------------------------------------------
10 */
11
12 /*
13 * PLEASE don't put C++ comments here - this is a C source file.
14 */
15
16 #ifndef __GSOCKET_STANDALONE__
17 #include "wx/setup.h"
18 #include "wx/platform.h"
19 #endif
20
21 #if wxUSE_SOCKETS || defined(__GSOCKET_STANDALONE__)
22
23 #ifdef __DARWIN__
24 #include <CoreServices/CoreServices.h>
25
26 #else
27 #include <MacHeaders.c>
28 #define OTUNIXERRORS 1
29 #include <OpenTransport.h>
30 #include <OpenTransportProviders.h>
31 #include <OpenTptInternet.h>
32 #endif
33 #if TARGET_CARBON && !defined(OTAssert)
34 #define OTAssert( str , cond ) /* does not exists in Carbon */
35 #endif
36
37 #include <assert.h>
38 #include <errno.h>
39 #include <string.h>
40 #include <unistd.h>
41 #include <stdio.h>
42 #include <stdlib.h>
43 #include <stddef.h>
44 #include <ctype.h>
45 #include <utime.h>
46
47 /*
48 * INADDR_BROADCAST is identical to INADDR_NONE which is not defined
49 * on all unices. INADDR_BROADCAST should be fine to indicate an error.
50 */
51 #ifndef INADDR_BROADCAST
52 #define INADDR_BROADCAST 0xFFFFFFFFUL
53 #endif
54 #ifndef INADDR_NONE
55 #define INADDR_NONE INADDR_BROADCAST
56 #endif
57 #ifndef INADDR_ANY
58 #define INADDR_ANY 0x0UL
59 #endif
60 #ifndef __GSOCKET_STANDALONE__
61
62 #include "wx/mac/macnotfy.h"
63 #include "wx/mac/gsockmac.h"
64 #include "wx/gsocket.h"
65
66 #else
67
68 #include "gsockmac.h"
69 #include "gsocket.h"
70
71 #endif /* __GSOCKET_STANDALONE__ */
72
73 #ifndef ntohl
74 #define ntohl(x) (x)
75 #define ntohs(x) (x)
76 #define htonl(x) (x)
77 #define htons(x) (x)
78 #endif
79
80 void wxCYield() ;
81 #ifdef __WXDEBUG__
82 #define qDebug 1
83 #define qDebug2 1
84 extern pascal void OTDebugStr(const char* str);
85 #endif
86 #ifndef __DARWIN__
87 #include <OTDebug.h>
88 #endif
89 InetSvcRef gInetSvcRef = 0 ;
90 int gOTInited = 0 ;
91 OTNotifyUPP gOTNotifierUPP = NULL ;
92
93 OSStatus DoNegotiateIPReuseAddrOption(EndpointRef ep, Boolean enableReuseIPMode);
94
95 /* Input: ep - endpointref on which to negotiate the option
96 enableReuseIPMode - desired option setting - true/false
97 Return: kOTNoError indicates that the option was successfully negotiated
98 OSStatus is an error if < 0, otherwise, the status field is
99 returned and is > 0.
100
101 IMPORTANT NOTE: The endpoint is assumed to be in synchronous more, otherwise
102 this code will not function as desired
103 */
104
105
106 OSStatus DoNegotiateIPReuseAddrOption(EndpointRef ep, Boolean enableReuseIPMode)
107
108 {
109 UInt8 buf[kOTFourByteOptionSize]; // define buffer for fourByte Option size
110 TOption* opt; // option ptr to make items easier to access
111 TOptMgmt req;
112 TOptMgmt ret;
113 OSStatus err;
114
115 if (!OTIsSynchronous(ep))
116 {
117 return (-1);
118 }
119 opt = (TOption*)buf; // set option ptr to buffer
120 req.opt.buf = buf;
121 req.opt.len = sizeof(buf);
122 req.flags = T_NEGOTIATE; // negotiate for option
123
124 ret.opt.buf = buf;
125 ret.opt.maxlen = kOTFourByteOptionSize;
126
127 opt->level = INET_IP; // dealing with an IP Level function
128 #ifdef __DARWIN__
129 opt->name = kIP_REUSEADDR;
130 #else
131 opt->name = IP_REUSEADDR;
132 #endif
133 opt->len = kOTFourByteOptionSize;
134 opt->status = 0;
135 *(UInt32*)opt->value = enableReuseIPMode; // set the desired option level, true or false
136
137 err = OTOptionManagement(ep, &req, &ret);
138
139 // if no error then return the option status value
140 if (err == kOTNoError)
141 {
142 if (opt->status != T_SUCCESS)
143 err = opt->status;
144 else
145 err = kOTNoError;
146 }
147
148 return err;
149 }
150
151
152 pascal void OTInetEventHandler(void*s, OTEventCode event, OTResult, void *cookie) ;
153 pascal void OTInetEventHandler(void*s, OTEventCode event, OTResult result, void *cookie)
154 {
155 int wakeUp = true ;
156 GSocket* sock = (GSocket*) s ;
157
158 if ( event == kOTSyncIdleEvent )
159 {
160 return ;
161 }
162
163 if ( s )
164 {
165 wxMacAddEvent( sock->m_mac_events , _GSocket_Internal_Proc , event , s , wakeUp ) ;
166 }
167
168 return;
169 }
170
171 static void SetDefaultEndpointModes(EndpointRef ep , void *data )
172 // This routine sets the supplied endpoint into the default
173 // mode used in this application. The specifics are:
174 // blocking, synchronous, and using synch idle events with
175 // the standard YieldingNotifier.
176 {
177 OSStatus junk = kOTNoError ;
178 OTAssert ("SetDefaultEndpointModes:invalid ref", ep != kOTInvalidEndpointRef ) ;
179 junk = OTSetAsynchronous(ep);
180 OTAssert("SetDefaultEndpointModes: Could not set asynchronous", junk == noErr);
181 /*
182 junk = OTSetBlocking(ep);
183 OTAssert("SetDefaultEndpointModes: Could not set blocking", junk == noErr);
184 junk = OTSetSynchronous(ep);
185 OTAssert("SetDefaultEndpointModes: Could not set synchronous", junk == noErr);
186 junk = OTSetBlocking(ep);
187 OTAssert("SetDefaultEndpointModes: Could not set blocking", junk == noErr);
188 */
189 junk = OTInstallNotifier(ep, gOTNotifierUPP, data);
190 OTAssert("SetDefaultEndpointModes: Could not install notifier", junk == noErr);
191 /*
192 junk = OTUseSyncIdleEvents(ep, true);
193 OTAssert("SetDefaultEndpointModes: Could not use sync idle events", junk == noErr);
194 */
195 }
196
197 /* Global initialisers */
198
199 void GSocket_SetGUIFunctions(GSocketGUIFunctionsTable *table)
200 {
201 // do nothing, wxMac doesn't have wxBase-GUI separation yet
202 }
203
204 int GSocket_Init()
205 {
206 return 1;
207 }
208
209 bool GSocket_Verify_Inited() ;
210 bool GSocket_Verify_Inited()
211 {
212 OSStatus err ;
213 #if TARGET_CARBON
214 // Marc Newsam: added the clientcontext variable
215 // however, documentation is unclear how this works
216 OTClientContextPtr clientcontext;
217
218 if ( gInetSvcRef )
219 return true ;
220
221 InitOpenTransportInContext(kInitOTForApplicationMask, &clientcontext);
222 gOTInited = 1 ;
223 gInetSvcRef = OTOpenInternetServicesInContext(kDefaultInternetServicesPath,
224 NULL, &err, clientcontext);
225 #else
226 if ( gInetSvcRef )
227 return true ;
228
229 InitOpenTransport() ;
230 gOTInited = 1 ;
231 gInetSvcRef = OTOpenInternetServices(kDefaultInternetServicesPath, NULL, &err);
232 #endif
233 if ( gInetSvcRef == NULL || err != kOTNoError )
234 {
235 OTAssert("Could not open Inet Services", err == noErr);
236 return false ;
237 }
238 gOTNotifierUPP = NewOTNotifyUPP( OTInetEventHandler ) ;
239 return true ;
240 }
241
242 void GSocket_Cleanup()
243 {
244 if ( gOTInited != 0 )
245 {
246 if ( gInetSvcRef != NULL )
247 OTCloseProvider( gInetSvcRef );
248 #if TARGET_CARBON
249 CloseOpenTransportInContext( NULL ) ;
250 #else
251 CloseOpenTransport() ;
252 #endif
253 if ( gOTNotifierUPP )
254 DisposeOTNotifyUPP( gOTNotifierUPP ) ;
255 }
256 }
257
258 /* Constructors / Destructors for GSocket */
259
260 GSocket::GSocket()
261 {
262 int i;
263
264 m_ok = GSocket_Verify_Inited();
265
266 m_endpoint = NULL ;
267 for (i=0;i<GSOCK_MAX_EVENT;i++)
268 {
269 m_cbacks[i] = NULL;
270 }
271 m_detected = 0;
272 m_local = NULL;
273 m_peer = NULL;
274 m_error = GSOCK_NOERROR;
275 m_server = false;
276 m_stream = true;
277 m_non_blocking = false;
278 m_timeout = 1*1000;
279 /* 10 sec * 1000 millisec */
280 m_takesEvents = true ;
281 m_mac_events = wxMacGetNotifierTable() ;
282 }
283
284 GSocket::~GSocket()
285 {
286 assert(this);
287
288 /* Check that the socket is really shutdowned */
289 if (m_endpoint != kOTInvalidEndpointRef)
290 Shutdown();
291
292
293 /* Destroy private addresses */
294 if (m_local)
295 GAddress_destroy(m_local);
296
297 if (m_peer)
298 GAddress_destroy(m_peer);
299 }
300
301 /* GSocket_Shutdown:
302 * Disallow further read/write operations on this socket, close
303 * the fd and disable all callbacks.
304 */
305 void GSocket::Shutdown()
306 {
307 OSStatus err ;
308 int evt;
309
310 assert(this);
311
312 /* If socket has been created, shutdown it */
313 if (m_endpoint != kOTInvalidEndpointRef )
314 {
315 err = OTSndOrderlyDisconnect( m_endpoint ) ;
316 if ( err != kOTNoError )
317 {
318
319 }
320 err = OTRcvOrderlyDisconnect( m_endpoint ) ;
321 err = OTUnbind( m_endpoint ) ;
322 err = OTCloseProvider( m_endpoint ) ;
323 m_endpoint = kOTInvalidEndpointRef ;
324 }
325
326 /* Disable GUI callbacks */
327 for (evt = 0; evt < GSOCK_MAX_EVENT; evt++)
328 m_cbacks[evt] = NULL;
329
330 m_detected = 0;
331 Disable_Events();
332 wxMacRemoveAllNotifiersForData( wxMacGetNotifierTable() , this ) ;
333 }
334
335
336 /* Address handling */
337
338 /* GSocket_SetLocal:
339 * GSocket_GetLocal:
340 * GSocket_SetPeer:
341 * GSocket_GetPeer:
342 * Set or get the local or peer address for this socket. The 'set'
343 * functions return GSOCK_NOERROR on success, an error code otherwise.
344 * The 'get' functions return a pointer to a GAddress object on success,
345 * or NULL otherwise, in which case they set the error code of the
346 * corresponding GSocket.
347 *
348 * Error codes:
349 * GSOCK_INVSOCK - the socket is not valid.
350 * GSOCK_INVADDR - the address is not valid.
351 */
352 GSocketError GSocket::SetLocal(GAddress *address)
353 {
354 assert(this);
355
356 /* the socket must be initialized, or it must be a server */
357 if ((m_endpoint != kOTInvalidEndpointRef && !m_server))
358 {
359 m_error = GSOCK_INVSOCK;
360 return GSOCK_INVSOCK;
361 }
362
363 /* check address */
364 if (address == NULL || address->m_family == GSOCK_NOFAMILY)
365 {
366 m_error = GSOCK_INVADDR;
367 return GSOCK_INVADDR;
368 }
369
370 if (m_local)
371 GAddress_destroy(m_local);
372
373 m_local = GAddress_copy(address);
374
375 return GSOCK_NOERROR;
376 }
377
378 GSocketError GSocket::SetPeer(GAddress *address)
379 {
380 assert(this);
381
382 /* check address */
383 if (address == NULL || address->m_family == GSOCK_NOFAMILY)
384 {
385 m_error = GSOCK_INVADDR;
386 return GSOCK_INVADDR;
387 }
388
389 if (m_peer)
390 GAddress_destroy(m_peer);
391
392 m_peer = GAddress_copy(address);
393
394 return GSOCK_NOERROR;
395 }
396
397 GAddress *GSocket::GetLocal()
398 {
399 GAddress *address = NULL ;
400 GSocketError err;
401 InetAddress loc ;
402
403 assert(this);
404
405 /* try to get it from the m_local var first */
406 if (m_local)
407 return GAddress_copy(m_local);
408
409 /* else, if the socket is initialized, try getsockname */
410 if (m_endpoint == kOTInvalidEndpointRef)
411 {
412 m_error = GSOCK_INVSOCK;
413 return NULL;
414 }
415
416
417 /* we do not support multihoming with this code at the moment
418 OTGetProtAddress will have to be used then - but we don't have a handy
419 method to use right now
420 */
421 {
422 InetInterfaceInfo info;
423 OTInetGetInterfaceInfo(&info, kDefaultInetInterface);
424 loc.fHost = info.fAddress ;
425 loc.fPort = 0 ;
426 loc.fAddressType = AF_INET ;
427 }
428
429 /* got a valid address from getsockname, create a GAddress object */
430 address = GAddress_new();
431 if (address == NULL)
432 {
433 m_error = GSOCK_MEMERR;
434 return NULL;
435 }
436
437 err = _GAddress_translate_from(address, &loc);
438 if (err != GSOCK_NOERROR)
439 {
440 GAddress_destroy(address);
441 m_error = err;
442 return NULL;
443 }
444
445 return address;
446 }
447
448 GAddress *GSocket::GetPeer()
449 {
450 assert(this);
451
452 /* try to get it from the m_peer var */
453 if (m_peer)
454 return GAddress_copy(m_peer);
455
456 return NULL;
457 }
458
459 /* Server specific parts */
460
461 /* GSocket_SetServer:
462 * Sets up this socket as a server. The local address must have been
463 * set with GSocket_SetLocal() before GSocket_SetServer() is called.
464 * Returns GSOCK_NOERROR on success, one of the following otherwise:
465 *
466 * Error codes:
467 * GSOCK_INVSOCK - the socket is in use.
468 * GSOCK_INVADDR - the local address has not been set.
469 * GSOCK_IOERR - low-level error.
470 */
471 GSocketError GSocket::SetServer()
472 {
473 assert(this);
474
475 /* must not be in use */
476 if (m_endpoint != kOTInvalidEndpointRef )
477 {
478 m_error = GSOCK_INVSOCK;
479 return GSOCK_INVSOCK;
480 }
481
482 /* the local addr must have been set */
483 if (!m_local)
484 {
485 m_error = GSOCK_INVADDR;
486 return GSOCK_INVADDR;
487 }
488
489 /* Initialize all fields */
490 m_stream = true;
491 m_server = true;
492 m_oriented = true;
493
494 // TODO
495 #if 0
496 /* Create the socket */
497 m_endpoint = socket(m_local->m_realfamily, SOCK_STREAM, 0);
498 socket_set_ref( m_endpoint , (unsigned long) &gMacNetEvents , (unsigned long) this ) ;
499 if (m_endpoint == kOTInvalidEndpointRef)
500 {
501 m_error = GSOCK_IOERR;
502 return GSOCK_IOERR;
503 }
504
505 ioctl(m_endpoint, FIONBIO, &arg);
506 Enable_Events();
507
508 /* Bind to the local address,
509 * retrieve the actual address bound,
510 * and listen up to 5 connections.
511 */
512 if ((bind(m_endpoint, m_local->m_addr, m_local->m_len) != 0) ||
513 (getsockname(m_endpoint,
514 m_local->m_addr,
515 (SOCKLEN_T *) &m_local->m_len) != 0) ||
516 (listen(m_endpoint, 5) != 0))
517 {
518 close(m_endpoint);
519 m_endpoint = -1;
520 m_error = GSOCK_IOERR;
521 return GSOCK_IOERR;
522 }
523 #endif
524 return GSOCK_NOERROR;
525 }
526
527 /* GSocket_WaitConnection:
528 * Waits for an incoming client connection. Returns a pointer to
529 * a GSocket object, or NULL if there was an error, in which case
530 * the last error field will be updated for the calling GSocket.
531 *
532 * Error codes (set in the calling GSocket)
533 * GSOCK_INVSOCK - the socket is not valid or not a server.
534 * GSOCK_TIMEDOUT - timeout, no incoming connections.
535 * GSOCK_WOULDBLOCK - the call would block and the socket is nonblocking.
536 * GSOCK_MEMERR - couldn't allocate memory.
537 * GSOCK_IOERR - low-level error.
538 */
539 GSocket *GSocket::WaitConnection()
540 {
541 GSocket *connection = NULL ;
542
543 assert(this);
544
545 /* Reenable CONNECTION events */
546 m_detected &= ~GSOCK_CONNECTION_FLAG;
547
548 /* If the socket has already been created, we exit immediately */
549 if (m_endpoint == kOTInvalidEndpointRef || !m_server)
550 {
551 m_error = GSOCK_INVSOCK;
552 return NULL;
553 }
554
555 /* Create a GSocket object for the new connection */
556 connection = GSocket_new();
557
558 if (!connection)
559 {
560 m_error = GSOCK_MEMERR;
561 return NULL;
562 }
563
564 /* Wait for a connection (with timeout) */
565 if (Input_Timeout() == GSOCK_TIMEDOUT)
566 {
567 delete connection;
568 /* m_error set by _GSocket_Input_Timeout */
569 return NULL;
570 }
571
572 // TODO
573 #if 0
574 connection->m_endpoint = accept(m_endpoint, &from, (SOCKLEN_T *) &fromlen);
575 #endif
576
577 if (connection->m_endpoint == kOTInvalidEndpointRef )
578 {
579 if (errno == EWOULDBLOCK)
580 m_error = GSOCK_WOULDBLOCK;
581 else
582 m_error = GSOCK_IOERR;
583
584 delete connection;
585 return NULL;
586 }
587
588 /* Initialize all fields */
589 connection->m_server = false;
590 connection->m_stream = true;
591 connection->m_oriented = true;
592
593 /* Setup the peer address field */
594 connection->m_peer = GAddress_new();
595 if (!connection->m_peer)
596 {
597 delete connection;
598 m_error = GSOCK_MEMERR;
599 return NULL;
600 }
601 // TODO
602 #if 0
603 err = _GAddress_translate_from(connection->m_peer, &from, fromlen);
604 if (err != GSOCK_NOERROR)
605 {
606 GAddress_destroy(connection->m_peer);
607 GSocket_destroy(connection);
608 m_error = err;
609 return NULL;
610 }
611
612 ioctl(connection->m_endpoint, FIONBIO, &arg);
613 #endif
614 connection->Enable_Events();
615
616 return connection;
617 }
618
619 /* Datagram sockets */
620
621 /* GSocket_SetNonOriented:
622 * Sets up this socket as a non-connection oriented (datagram) socket.
623 * Before using this function, the local address must have been set
624 * with GSocket_SetLocal(), or the call will fail. Returns GSOCK_NOERROR
625 * on success, or one of the following otherwise.
626 *
627 * Error codes:
628 * GSOCK_INVSOCK - the socket is in use.
629 * GSOCK_INVADDR - the local address has not been set.
630 * GSOCK_IOERR - low-level error.
631 */
632 GSocketError GSocket::SetNonOriented()
633 {
634 assert(this);
635
636 if (m_endpoint != kOTInvalidEndpointRef )
637 {
638 m_error = GSOCK_INVSOCK;
639 return GSOCK_INVSOCK;
640 }
641
642 if (!m_local)
643 {
644 m_error = GSOCK_INVADDR;
645 return GSOCK_INVADDR;
646 }
647
648 /* Initialize all fields */
649 m_stream = false;
650 m_server = false;
651 m_oriented = false;
652
653 /* Create the socket */
654
655 // TODO
656 #if 0
657 m_endpoint = socket(m_local->m_realfamily, SOCK_DGRAM, 0);
658 socket_set_ref( m_endpoint , (unsigned long) &gMacNetEvents , (unsigned long) this ) ;
659 #endif
660 if (m_endpoint == kOTInvalidEndpointRef )
661 {
662 m_error = GSOCK_IOERR;
663 return GSOCK_IOERR;
664 }
665
666 // TODO
667 #if 0
668 ioctl(m_endpoint, FIONBIO, &arg);
669 #endif
670 Enable_Events();
671
672 /* Bind to the local address,
673 * and retrieve the actual address bound.
674 */
675 // TODO
676 #if 0
677 if ((bind(m_endpoint, m_local->m_addr, m_local->m_len) != 0) ||
678 (getsockname(m_endpoint,
679 m_local->m_addr,
680 (SOCKLEN_T *) &m_local->m_len) != 0))
681 {
682 close(m_endpoint);
683 m_endpoint = -1;
684 m_error = GSOCK_IOERR;
685 return GSOCK_IOERR;
686 }
687 #endif
688 return GSOCK_NOERROR;
689 }
690
691 /* Client specific parts */
692
693 /* GSocket_Connect:
694 * For stream (connection oriented) sockets, GSocket_Connect() tries
695 * to establish a client connection to a server using the peer address
696 * as established with GSocket_SetPeer(). Returns GSOCK_NOERROR if the
697 * connection has been succesfully established, or one of the error
698 * codes listed below. Note that for nonblocking sockets, a return
699 * value of GSOCK_WOULDBLOCK doesn't mean a failure. The connection
700 * request can be completed later; you should use GSocket_Select()
701 * to poll for GSOCK_CONNECTION | GSOCK_LOST, or wait for the
702 * corresponding asynchronous events.
703 *
704 * For datagram (non connection oriented) sockets, GSocket_Connect()
705 * just sets the peer address established with GSocket_SetPeer() as
706 * default destination.
707 *
708 * Error codes:
709 * GSOCK_INVSOCK - the socket is in use or not valid.
710 * GSOCK_INVADDR - the peer address has not been established.
711 * GSOCK_TIMEDOUT - timeout, the connection failed.
712 * GSOCK_WOULDBLOCK - connection in progress (nonblocking sockets only)
713 * GSOCK_MEMERR - couldn't allocate memory.
714 * GSOCK_IOERR - low-level error.
715 */
716 GSocketError GSocket::Connect(GSocketStream stream)
717 {
718 InetAddress addr ;
719 TEndpointInfo info;
720 OSStatus err = kOTNoError;
721 TCall peer ;
722
723 assert(this);
724
725 /* Enable CONNECTION events (needed for nonblocking connections) */
726 m_detected &= ~GSOCK_CONNECTION_FLAG;
727
728 if (m_endpoint != kOTInvalidEndpointRef )
729 {
730 m_error = GSOCK_INVSOCK;
731 return GSOCK_INVSOCK;
732 }
733
734 if (!m_peer)
735 {
736 m_error = GSOCK_INVADDR;
737 return GSOCK_INVADDR;
738 }
739
740 /* Streamed or dgram socket? */
741 m_stream = (stream == GSOCK_STREAMED);
742 m_oriented = true;
743 m_server = false;
744
745 /* Create the socket */
746 #if TARGET_CARBON
747 m_endpoint =
748 OTOpenEndpointInContext( OTCreateConfiguration( kTCPName) , 0 , &info , &err , NULL ) ;
749 #else
750 m_endpoint =
751 OTOpenEndpoint( OTCreateConfiguration( kTCPName) , 0 , &info , &err ) ;
752 #endif
753 if ( m_endpoint == kOTInvalidEndpointRef || err != kOTNoError )
754 {
755 m_endpoint = kOTInvalidEndpointRef ;
756 m_error = GSOCK_IOERR;
757 return GSOCK_IOERR;
758 }
759 err = OTBind( m_endpoint , nil , nil ) ;
760 if ( err != kOTNoError )
761 {
762 return GSOCK_IOERR;
763 }
764 SetDefaultEndpointModes( m_endpoint , this ) ;
765 // TODO
766 #if 0
767 ioctl(m_endpoint, FIONBIO, &arg);
768 #endif
769 Enable_Events();
770
771 _GAddress_translate_to( m_peer , &addr ) ;
772 memset( &peer , 0 , sizeof( TCall ) ) ;
773 peer.addr.len = sizeof( InetAddress ) ;
774 peer.addr.buf = (unsigned char*) &addr ;
775 err = OTConnect( m_endpoint , &peer , nil ) ;
776 if ( err != noErr )
777 {
778 /* If connect failed with EINPROGRESS and the GSocket object
779 * is in blocking mode, we select() for the specified timeout
780 * checking for writability to see if the connection request
781 * completes.
782 */
783
784 if ((err == kOTNoDataErr ) && (!m_non_blocking))
785 {
786 if (Output_Timeout() == GSOCK_TIMEDOUT)
787 {
788 OTSndOrderlyDisconnect( m_endpoint ) ;
789 m_endpoint = kOTInvalidEndpointRef ;
790 /* m_error is set in _GSocket_Output_Timeout */
791 return GSOCK_TIMEDOUT;
792 }
793 else
794 {
795 /*
796 int error;
797 SOCKLEN_T len = sizeof(error);
798
799 getsockopt(m_endpoint, SOL_SOCKET, SO_ERROR, (void*) &error, &len);
800
801 if (!error)
802 */
803 return GSOCK_NOERROR;
804 }
805 }
806
807 /* If connect failed with EINPROGRESS and the GSocket object
808 * is set to nonblocking, we set m_error to GSOCK_WOULDBLOCK
809 * (and return GSOCK_WOULDBLOCK) but we don't close the socket;
810 * this way if the connection completes, a GSOCK_CONNECTION
811 * event will be generated, if enabled.
812 */
813 if ((err == kOTNoDataErr) && (m_non_blocking))
814 {
815 m_error = GSOCK_WOULDBLOCK;
816 return GSOCK_WOULDBLOCK;
817 }
818
819 /* If connect failed with an error other than EINPROGRESS,
820 * then the call to GSocket_Connect has failed.
821 */
822 OTSndOrderlyDisconnect( m_endpoint ) ;
823
824 m_endpoint = kOTInvalidEndpointRef ;
825 m_error = GSOCK_IOERR;
826 return GSOCK_IOERR;
827 }
828 // OTInetEventHandler(this, T_CONNECT , kOTNoError , NULL ) ;
829 return GSOCK_NOERROR;
830 }
831
832 /* Generic IO */
833
834 /* Like recv(), send(), ... */
835 int GSocket::Read(char *buffer, int size)
836 {
837 int ret = 0 ;
838
839 assert(this);
840
841 /* Reenable INPUT events */
842 m_detected &= ~GSOCK_INPUT_FLAG;
843
844 if (m_endpoint == kOTInvalidEndpointRef || m_server)
845 {
846 m_error = GSOCK_INVSOCK;
847 return -1;
848 }
849
850 /* If the socket is blocking, wait for data (with a timeout) */
851 if (Input_Timeout() == GSOCK_TIMEDOUT)
852 return -1;
853
854 /* Read the data */
855 if (m_stream)
856 ret = Recv_Stream(buffer, size);
857 else
858 ret = Recv_Dgram(buffer, size);
859
860 if (ret == -1)
861 {
862 if (errno == EWOULDBLOCK)
863 m_error = GSOCK_WOULDBLOCK;
864 else
865 m_error = GSOCK_IOERR;
866 }
867
868 return ret;
869 }
870
871 int GSocket::Write(const char *buffer, int size)
872 {
873 int ret;
874
875 assert(this);
876
877 if (m_endpoint == kOTInvalidEndpointRef || m_server)
878 {
879 m_error = GSOCK_INVSOCK;
880 return -1;
881 }
882
883 /* If the socket is blocking, wait for writability (with a timeout) */
884 if (Output_Timeout() == GSOCK_TIMEDOUT)
885 return -1;
886
887 /* Write the data */
888 if (m_stream)
889 ret = Send_Stream(buffer, size);
890 else
891 ret = Send_Dgram(buffer, size);
892
893 if (ret == -1)
894 {
895 if (errno == EWOULDBLOCK)
896 m_error = GSOCK_WOULDBLOCK;
897 else
898 m_error = GSOCK_IOERR;
899
900 /* Only reenable OUTPUT events after an error (just like WSAAsyncSelect
901 * in MSW). Once the first OUTPUT event is received, users can assume
902 * that the socket is writable until a read operation fails. Only then
903 * will further OUTPUT events be posted.
904 */
905 m_detected &= ~GSOCK_OUTPUT_FLAG;
906 return -1;
907 }
908
909 return ret;
910 }
911
912 /* GSocket_Select:
913 * Polls the socket to determine its status. This function will
914 * check for the events specified in the 'flags' parameter, and
915 * it will return a mask indicating which operations can be
916 * performed. This function won't block, regardless of the
917 * mode (blocking | nonblocking) of the socket.
918 */
919 GSocketEventFlags GSocket::Select(GSocketEventFlags flags)
920 {
921 assert(this);
922 wxMacProcessNotifierEvents() ;
923 /*
924 state = OTGetEndpointState(m_endpoint);
925
926 if ( ( flags & GSOCK_INPUT_FLAG ) && ! ( m_detected & GSOCK_INPUT_FLAG ) )
927 {
928 size_t sz = 0 ;
929 OTCountDataBytes( m_endpoint , &sz ) ;
930 if ( state == T_INCON || sz > 0 )
931 {
932 m_detected |= GSOCK_INPUT_FLAG ;
933 (m_cbacks[GSOCK_INPUT])(this, GSOCK_INPUT, m_data[GSOCK_INPUT]);
934 }
935 }
936 if ( ( flags & GSOCK_INPUT_FLAG ) && ! ( m_detected & GSOCK_OUTPUT_FLAG ) )
937 {
938 if ( state == T_DATAXFER || state == T_INREL )
939 {
940 m_detected |=GSOCK_OUTPUT_FLAG ;
941 (m_cbacks[GSOCK_OUTPUT])(this, GSOCK_OUTPUT, m_data[GSOCK_OUTPUT]);
942 }
943 }
944 */
945 return ( flags & m_detected ) ;
946 }
947
948 /* Flags */
949
950 /* GSocket_SetNonBlocking:
951 * Sets the socket to non-blocking mode. All IO calls will return
952 * immediately.
953 */
954 void GSocket::SetNonBlocking(bool non_block)
955 {
956 assert(this);
957
958 m_non_blocking = non_block;
959 }
960
961 /* GSocket_SetTimeout:
962 * Sets the timeout for blocking calls. Time is expressed in
963 * milliseconds.
964 */
965 void GSocket::SetTimeout(unsigned long millisec)
966 {
967 assert(this);
968
969 // this is usually set too high and we have not yet been able to detect a closed
970 // stream, thus we leave the 10 sec timeout
971 // m_timeout = millisec;
972 }
973
974 /* GSocket_GetError:
975 * Returns the last error occured for this socket. Note that successful
976 * operations do not clear this back to GSOCK_NOERROR, so use it only
977 * after an error.
978 */
979 GSocketError WXDLLIMPEXP_NET GSocket::GetError()
980 {
981 assert(this);
982
983 return m_error;
984 }
985
986 /* Callbacks */
987
988 /* GSOCK_INPUT:
989 * There is data to be read in the input buffer. If, after a read
990 * operation, there is still data available, the callback function will
991 * be called again.
992 * GSOCK_OUTPUT:
993 * The socket is available for writing. That is, the next write call
994 * won't block. This event is generated only once, when the connection is
995 * first established, and then only if a call failed with GSOCK_WOULDBLOCK,
996 * when the output buffer empties again. This means that the app should
997 * assume that it can write since the first OUTPUT event, and no more
998 * OUTPUT events will be generated unless an error occurs.
999 * GSOCK_CONNECTION:
1000 * Connection succesfully established, for client sockets, or incoming
1001 * client connection, for server sockets. Wait for this event (also watch
1002 * out for GSOCK_LOST) after you issue a nonblocking GSocket_Connect() call.
1003 * GSOCK_LOST:
1004 * The connection is lost (or a connection request failed); this could
1005 * be due to a failure, or due to the peer closing it gracefully.
1006 */
1007
1008 /* GSocket_SetCallback:
1009 * Enables the callbacks specified by 'flags'. Note that 'flags'
1010 * may be a combination of flags OR'ed toghether, so the same
1011 * callback function can be made to accept different events.
1012 * The callback function must have the following prototype:
1013 *
1014 * void function(GSocket *socket, GSocketEvent event, char *cdata)
1015 */
1016 void GSocket::SetCallback(GSocketEventFlags flags,
1017 GSocketCallback callback, char *cdata)
1018 {
1019 int count;
1020
1021 assert(this);
1022
1023 for (count = 0; count < GSOCK_MAX_EVENT; count++)
1024 {
1025 if ((flags & (1 << count)) != 0)
1026 {
1027 m_cbacks[count] = callback;
1028 m_data[count] = cdata;
1029 }
1030 }
1031 }
1032
1033 /* GSocket_UnsetCallback:
1034 * Disables all callbacks specified by 'flags', which may be a
1035 * combination of flags OR'ed toghether.
1036 */
1037 void GSocket::UnsetCallback(GSocketEventFlags flags)
1038 {
1039 int count;
1040
1041 assert(this);
1042
1043 for (count = 0; count < GSOCK_MAX_EVENT; count++)
1044 {
1045 if ((flags & (1 << count)) != 0)
1046 {
1047 m_cbacks[count] = NULL;
1048 m_data[count] = NULL;
1049 }
1050 }
1051 }
1052
1053
1054 #define CALL_CALLBACK(socket, event) { \
1055 _GSocket_Disable(socket, event); \
1056 if (socket->m_cbacks[event]) \
1057 socket->m_cbacks[event](socket, event, socket->m_data[event]); \
1058 }
1059
1060 int GSocket::Recv_Stream(char *buffer, int size)
1061 {
1062 OTFlags flags ;
1063 OTResult res ;
1064 OTByteCount sz = 0 ;
1065
1066 OTCountDataBytes( m_endpoint , &sz ) ;
1067 if ( size > (int)sz )
1068 size = sz ;
1069 res = OTRcv( m_endpoint , buffer , size , &flags ) ;
1070 if ( res < 0 )
1071 {
1072 return -1 ;
1073 }
1074
1075 // we simulate another read event if there are still bytes
1076 if ( m_takesEvents )
1077 {
1078 OTByteCount sz = 0 ;
1079 OTCountDataBytes( m_endpoint , &sz ) ;
1080 if ( sz > 0 )
1081 {
1082 m_detected |= GSOCK_INPUT_FLAG ;
1083 (m_cbacks[GSOCK_INPUT])(this, GSOCK_INPUT, m_data[GSOCK_INPUT]);
1084 }
1085 }
1086 return res ;
1087 }
1088
1089 int GSocket::Recv_Dgram(char *buffer, int size)
1090 {
1091 // TODO
1092 int ret = -1;
1093 #if 0
1094 struct sockaddr from;
1095 SOCKLEN_T fromlen = sizeof(from);
1096 GSocketError err;
1097
1098 fromlen = sizeof(from);
1099
1100 ret = recvfrom(m_endpoint, buffer, size, 0, &from, (SOCKLEN_T *) &fromlen);
1101
1102 if (ret == -1)
1103 return -1;
1104
1105 /* Translate a system address into a GSocket address */
1106 if (!m_peer)
1107 {
1108 m_peer = GAddress_new();
1109 if (!m_peer)
1110 {
1111 m_error = GSOCK_MEMERR;
1112 return -1;
1113 }
1114 }
1115 err = _GAddress_translate_from(m_peer, &from, fromlen);
1116 if (err != GSOCK_NOERROR)
1117 {
1118 GAddress_destroy(m_peer);
1119 m_peer = NULL;
1120 m_error = err;
1121 return -1;
1122 }
1123 #endif
1124 return ret;
1125 }
1126
1127 int GSocket::Send_Stream(const char *buffer, int size)
1128 {
1129 OTFlags flags = 0 ;
1130 OTResult res ;
1131
1132 res = OTSnd( m_endpoint , (void*) buffer , size , flags ) ;
1133 return res ;
1134 }
1135
1136 int GSocket::Send_Dgram(const char *buffer, int size)
1137 {
1138 int ret = -1 ;
1139 // TODO
1140 #if 0
1141 struct sockaddr *addr;
1142 int len ;
1143 GSocketError err;
1144
1145 if (!m_peer)
1146 {
1147 m_error = GSOCK_INVADDR;
1148 return -1;
1149 }
1150
1151 err = _GAddress_translate_to(m_peer, &addr, &len);
1152 if (err != GSOCK_NOERROR)
1153 {
1154 m_error = err;
1155 return -1;
1156 }
1157
1158 ret = sendto(m_endpoint, buffer, size, 0, addr, len);
1159
1160 /* Frees memory allocated from _GAddress_translate_to */
1161 free(addr);
1162 #endif
1163 return ret;
1164 }
1165
1166 /* Compatibility functions for GSocket */
1167 GSocket *GSocket_new(void)
1168 {
1169 GSocket *newsocket = new GSocket();
1170 if(newsocket->IsOk())
1171 return newsocket;
1172 delete newsocket;
1173 return NULL;
1174 }
1175
1176
1177 /*
1178 * -------------------------------------------------------------------------
1179 * GAddress
1180 * -------------------------------------------------------------------------
1181 */
1182
1183 /* CHECK_ADDRESS verifies that the current family is either GSOCK_NOFAMILY
1184 * or GSOCK_*family*, and if it is GSOCK_NOFAMILY, it initalizes address
1185 * to be a GSOCK_*family*. In other cases, it returns GSOCK_INVADDR.
1186 */
1187 #define CHECK_ADDRESS(address, family, retval) \
1188 { \
1189 if (address->m_family == GSOCK_NOFAMILY) \
1190 if (_GAddress_Init_##family(address) != GSOCK_NOERROR) \
1191 return address->m_error; \
1192 if (address->m_family != GSOCK_##family) \
1193 { \
1194 address->m_error = GSOCK_INVADDR; \
1195 return retval; \
1196 } \
1197 }
1198
1199 GAddress *GAddress_new()
1200 {
1201 GAddress *address;
1202
1203 if ((address = (GAddress *) malloc(sizeof(GAddress))) == NULL)
1204 return NULL;
1205
1206 address->m_family = GSOCK_NOFAMILY;
1207 address->m_host = INADDR_NONE ;
1208 address->m_port = 0 ;
1209
1210 return address;
1211 }
1212
1213 GAddress *GAddress_copy(GAddress *address)
1214 {
1215 GAddress *addr2;
1216
1217 assert(address != NULL);
1218
1219 if ((addr2 = (GAddress *) malloc(sizeof(GAddress))) == NULL)
1220 return NULL;
1221
1222 memcpy(addr2, address, sizeof(GAddress));
1223 return addr2;
1224 }
1225
1226 void GAddress_destroy(GAddress *address)
1227 {
1228 assert(address != NULL);
1229
1230 free(address);
1231 }
1232
1233 void GAddress_SetFamily(GAddress *address, GAddressType type)
1234 {
1235 assert(address != NULL);
1236
1237 address->m_family = type;
1238 }
1239
1240 GAddressType GAddress_GetFamily(GAddress *address)
1241 {
1242 assert(address != NULL);
1243
1244 return address->m_family;
1245 }
1246
1247 GSocketError _GAddress_translate_from(GAddress *address,
1248 InetAddress *addr)
1249 {
1250 switch (addr->fAddressType)
1251 {
1252 case AF_INET:
1253 address->m_family = GSOCK_INET;
1254 break;
1255 #ifdef AF_INET6
1256 case AF_INET6:
1257 address->m_family = GSOCK_INET6;
1258 break;
1259 #endif
1260 default:
1261 {
1262 address->m_error = GSOCK_INVOP;
1263 return GSOCK_INVOP;
1264 }
1265 }
1266 address->m_host = addr->fHost ;
1267 address->m_port = addr->fPort ;
1268 return GSOCK_NOERROR;
1269 }
1270
1271 GSocketError _GAddress_translate_to(GAddress *address,
1272 InetAddress *addr)
1273 {
1274 if ( !GSocket_Verify_Inited() )
1275 return GSOCK_IOERR ;
1276 memset(addr, 0 , sizeof(struct InetAddress));
1277 OTInitInetAddress( addr , address->m_port , address->m_host ) ;
1278 return GSOCK_NOERROR;
1279 }
1280
1281 /*
1282 * -------------------------------------------------------------------------
1283 * Internet address family
1284 * -------------------------------------------------------------------------
1285 */
1286
1287 GSocketError _GAddress_Init_INET(GAddress *address)
1288 {
1289 address->m_family = GSOCK_INET;
1290 address->m_host = kOTAnyInetAddress ;
1291
1292 return GSOCK_NOERROR;
1293 }
1294
1295 GSocketError GAddress_INET_SetHostName(GAddress *address, const char *hostname)
1296 {
1297 InetHostInfo hinfo ;
1298 OSStatus ret ;
1299
1300 if ( !GSocket_Verify_Inited() )
1301 return GSOCK_IOERR ;
1302
1303 assert(address != NULL);
1304
1305 CHECK_ADDRESS(address, INET, GSOCK_INVADDR);
1306 ret = OTInetStringToAddress( gInetSvcRef , (char*) hostname , &hinfo ) ;
1307 if ( ret != kOTNoError )
1308 {
1309 address->m_host = INADDR_NONE ;
1310 address->m_error = GSOCK_NOHOST;
1311 return GSOCK_NOHOST;
1312 }
1313 address->m_host = hinfo.addrs[0] ;
1314 return GSOCK_NOERROR;
1315 }
1316
1317 GSocketError GAddress_INET_SetAnyAddress(GAddress *address)
1318 {
1319 return GAddress_INET_SetHostAddress(address, INADDR_ANY);
1320 }
1321
1322 GSocketError GAddress_INET_SetHostAddress(GAddress *address,
1323 unsigned long hostaddr)
1324 {
1325 assert(address != NULL);
1326
1327 CHECK_ADDRESS(address, INET, GSOCK_INVADDR);
1328
1329 address->m_host = htonl(hostaddr) ;
1330
1331 return GSOCK_NOERROR;
1332 }
1333
1334 struct service_entry
1335 {
1336 const char * name ;
1337 unsigned short port ;
1338 const char * protocol ;
1339 } ;
1340 typedef struct service_entry service_entry ;
1341
1342 service_entry gServices[] =
1343 {
1344 { "http" , 80 , "tcp" }
1345 } ;
1346
1347 GSocketError GAddress_INET_SetPortName(GAddress *address, const char *port,
1348 const char *protocol)
1349 {
1350 size_t i ;
1351
1352 assert(address != NULL);
1353 CHECK_ADDRESS(address, INET, GSOCK_INVADDR);
1354
1355 if (!port)
1356 {
1357 address->m_error = GSOCK_INVPORT;
1358 return GSOCK_INVPORT;
1359 }
1360 for ( i = 0 ; i < sizeof( gServices) / sizeof( service_entry ) ; ++i )
1361 {
1362 if ( strcmp( port , gServices[i].name ) == 0 )
1363 {
1364 if ( protocol == NULL || strcmp( protocol , gServices[i].protocol ) )
1365 {
1366 address->m_port = gServices[i].port ;
1367 return GSOCK_NOERROR;
1368 }
1369 }
1370 }
1371
1372 if (isdigit(port[0]))
1373 {
1374 address->m_port = atoi(port);
1375 return GSOCK_NOERROR;
1376 }
1377
1378 address->m_error = GSOCK_INVPORT;
1379 return GSOCK_INVPORT;
1380 }
1381
1382 GSocketError GAddress_INET_SetPort(GAddress *address, unsigned short port)
1383 {
1384 assert(address != NULL);
1385 CHECK_ADDRESS(address, INET, GSOCK_INVADDR);
1386 address->m_port = port ;
1387
1388 return GSOCK_NOERROR;
1389 }
1390
1391 GSocketError GAddress_INET_GetHostName(GAddress *address, char *hostname, size_t sbuf)
1392 {
1393 InetDomainName name ;
1394 if ( !GSocket_Verify_Inited() )
1395 return GSOCK_IOERR ;
1396
1397 assert(address != NULL);
1398 CHECK_ADDRESS(address, INET, GSOCK_INVADDR);
1399
1400 OTInetAddressToName( gInetSvcRef , address->m_host , name ) ;
1401 strncpy( hostname , name , sbuf ) ;
1402 return GSOCK_NOERROR;
1403 }
1404
1405 unsigned long GAddress_INET_GetHostAddress(GAddress *address)
1406 {
1407 assert(address != NULL);
1408 CHECK_ADDRESS(address, INET, 0);
1409
1410 return ntohl(address->m_host);
1411 }
1412
1413 unsigned short GAddress_INET_GetPort(GAddress *address)
1414 {
1415 assert(address != NULL);
1416 CHECK_ADDRESS(address, INET, 0);
1417
1418 return address->m_port;
1419 }
1420
1421 void GSocket::Enable_Events()
1422 {
1423 if ( m_takesEvents )
1424 return ;
1425
1426 {
1427 OTResult state ;
1428 m_takesEvents = true ;
1429 state = OTGetEndpointState(m_endpoint);
1430
1431 {
1432 OTByteCount sz = 0 ;
1433 OTCountDataBytes( m_endpoint , &sz ) ;
1434 if ( state == T_INCON || sz > 0 )
1435 {
1436 m_detected |= GSOCK_INPUT_FLAG ;
1437 (m_cbacks[GSOCK_INPUT])(this, GSOCK_INPUT, m_data[GSOCK_INPUT]);
1438 }
1439 }
1440 {
1441 if ( state == T_DATAXFER || state == T_INREL )
1442 {
1443 m_detected |=GSOCK_OUTPUT_FLAG ;
1444 (m_cbacks[GSOCK_OUTPUT])(this, GSOCK_OUTPUT, m_data[GSOCK_OUTPUT]);
1445 }
1446 }
1447 }
1448 }
1449
1450 void GSocket::Disable_Events()
1451 {
1452 m_takesEvents = false ;
1453 }
1454
1455 /* _GSocket_Input_Timeout:
1456 * For blocking sockets, wait until data is available or
1457 * until timeout ellapses.
1458 */
1459 GSocketError GSocket::Input_Timeout()
1460 {
1461 if ( !m_non_blocking )
1462 {
1463 UnsignedWide now , start ;
1464 bool formerTakesEvents = m_takesEvents ;
1465 Microseconds(&start);
1466 now = start ;
1467 m_takesEvents = false ;
1468
1469 while( (now.hi * 4294967296.0 + now.lo) - (start.hi * 4294967296.0 + start.lo) < m_timeout * 1000.0 )
1470 {
1471 OTResult state ;
1472 OTByteCount sz = 0 ;
1473 state = OTGetEndpointState(m_endpoint);
1474
1475 OTCountDataBytes( m_endpoint , &sz ) ;
1476 if ( state == T_INCON || sz > 0 )
1477 {
1478 m_takesEvents = formerTakesEvents ;
1479 return GSOCK_NOERROR;
1480 }
1481 Microseconds(&now);
1482 }
1483 m_takesEvents = formerTakesEvents ;
1484 m_error = GSOCK_TIMEDOUT;
1485 return GSOCK_TIMEDOUT;
1486 }
1487 return GSOCK_NOERROR;
1488 }
1489
1490 /* _GSocket_Output_Timeout:
1491 * For blocking sockets, wait until data can be sent without
1492 * blocking or until timeout ellapses.
1493 */
1494 GSocketError GSocket::Output_Timeout()
1495 {
1496 if ( !m_non_blocking )
1497 {
1498 UnsignedWide now , start ;
1499 bool formerTakesEvents = m_takesEvents ;
1500 Microseconds(&start);
1501 now = start ;
1502 m_takesEvents = false ;
1503
1504 while( (now.hi * 4294967296.0 + now.lo) - (start.hi * 4294967296.0 + start.lo) < m_timeout * 1000.0 )
1505 {
1506 OTResult state ;
1507 state = OTGetEndpointState(m_endpoint);
1508
1509 if ( state == T_DATAXFER || state == T_INREL )
1510 {
1511 m_takesEvents = formerTakesEvents ;
1512 return GSOCK_NOERROR;
1513 }
1514 Microseconds(&now);
1515 }
1516 m_takesEvents = formerTakesEvents ;
1517 m_error = GSOCK_TIMEDOUT;
1518 return GSOCK_TIMEDOUT;
1519 }
1520 return GSOCK_NOERROR;
1521 }
1522
1523 /* GSOCK_INPUT:
1524 * There is data to be read in the input buffer. If, after a read
1525 * operation, there is still data available, the callback function will
1526 * be called again.
1527 * GSOCK_OUTPUT:
1528 * The socket is available for writing. That is, the next write call
1529 * won't block. This event is generated only once, when the connection is
1530 * first established, and then only if a call failed with GSOCK_WOULDBLOCK,
1531 * when the output buffer empties again. This means that the app should
1532 * assume that it can write since the first OUTPUT event, and no more
1533 * OUTPUT events will be generated unless an error occurs.
1534 * GSOCK_CONNECTION:
1535 * Connection succesfully established, for client sockets, or incoming
1536 * client connection, for server sockets. Wait for this event (also watch
1537 * out for GSOCK_LOST) after you issue a nonblocking GSocket_Connect() call.
1538 * GSOCK_LOST:
1539 * The connection is lost (or a connection request failed); this could
1540 * be due to a failure, or due to the peer closing it gracefully.
1541 */
1542
1543 void _GSocket_Internal_Proc(unsigned long e , void* d )
1544 {
1545
1546 GSocket * socket = (GSocket*) d ;
1547 OTEventCode ev = (OTEventCode) e ;
1548 GSocketEvent event;
1549 GSocketEvent event2;
1550 GSocketCallback cback;
1551 char *data;
1552 GSocketCallback cback2;
1553 char *data2;
1554
1555 if ( !socket )
1556 return ;
1557 event = GSOCK_MAX_EVENT ;
1558 event2 = GSOCK_MAX_EVENT ;
1559 cback = NULL;
1560 data = NULL;
1561 cback2 = NULL;
1562 data2 = NULL;
1563
1564 /* Check that the socket still exists (it has not been
1565 * destroyed) and for safety, check that the m_endpoint field
1566 * is what we expect it to be.
1567 */
1568 if ((socket != NULL) && (socket->m_takesEvents))
1569 {
1570 switch (ev)
1571 {
1572 case T_LISTEN :
1573 event = GSOCK_CONNECTION ;
1574 break ;
1575 case T_CONNECT :
1576 event = GSOCK_CONNECTION ;
1577 event2 = GSOCK_OUTPUT ;
1578 {
1579 TCall retCall;
1580
1581 retCall.addr.buf = NULL;
1582 retCall.addr.maxlen = 0;
1583 retCall.opt.buf = NULL;
1584 retCall.opt.maxlen = 0;
1585 retCall.udata.buf = NULL;
1586 retCall.udata.maxlen= 0;
1587 OTRcvConnect( socket->m_endpoint , &retCall ) ;
1588 }
1589 break ;
1590 case T_DISCONNECT :
1591 event = GSOCK_LOST ;
1592 break ;
1593 case T_GODATA :
1594 case T_GOEXDATA :
1595 event = GSOCK_OUTPUT ;
1596 break ;
1597 case T_DATA :
1598 event = GSOCK_INPUT ;
1599 break ;
1600 case T_EXDATA :
1601 event = GSOCK_INPUT ;
1602 break ;
1603 }
1604 if (event != GSOCK_MAX_EVENT)
1605 {
1606 cback = socket->m_cbacks[event];
1607 data = socket->m_data[event];
1608
1609 if (event == GSOCK_LOST)
1610 socket->m_detected = GSOCK_LOST_FLAG;
1611 else
1612 socket->m_detected |= (1 << event);
1613 }
1614 if (event2 != GSOCK_MAX_EVENT)
1615 {
1616 cback2 = socket->m_cbacks[event2];
1617 data2 = socket->m_data[event2];
1618
1619 if (event2 == GSOCK_LOST)
1620 socket->m_detected = GSOCK_LOST_FLAG;
1621 else
1622 socket->m_detected |= (1 << event2);
1623 }
1624 }
1625
1626 /* OK, we can now leave the critical section because we have
1627 * already obtained the callback address (we make no further
1628 * accesses to socket->whatever). However, the app should
1629 * be prepared to handle events from a socket that has just
1630 * been closed!
1631 */
1632
1633 if (cback != NULL)
1634 (cback)(socket, event, data);
1635 if (cback2 != NULL)
1636 (cback2)(socket, event2, data2);
1637
1638 }
1639
1640 /* Hack added for Mac OS X */
1641 GSocketError GAddress_UNIX_GetPath(GAddress *addr, char *path, size_t buf)
1642 {
1643 return GSOCK_INVADDR;
1644 }
1645
1646 GSocketError GAddress_UNIX_SetPath(GAddress *addr, const char *path)
1647 {
1648 return GSOCK_INVADDR;
1649 }
1650
1651 #endif /* wxUSE_SOCKETS || defined(__GSOCKET_STANDALONE__) */