]> git.saurik.com Git - wxWidgets.git/blob - src/common/socket.cpp
266e81cce733918404b4ad56bc0cc2cdb167d62c
[wxWidgets.git] / src / common / socket.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/socket.cpp
3 // Purpose: Socket handler classes
4 // Authors: Guilhem Lavaux, Guillermo Rodriguez Garcia
5 // Created: April 1997
6 // Copyright: (C) 1999-1997, Guilhem Lavaux
7 // (C) 2000-1999, Guillermo Rodriguez Garcia
8 // RCS_ID: $Id$
9 // License: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
11
12 // ==========================================================================
13 // Declarations
14 // ==========================================================================
15
16 // For compilers that support precompilation, includes "wx.h".
17 #include "wx/wxprec.h"
18
19 #ifdef __BORLANDC__
20 #pragma hdrstop
21 #endif
22
23 #if wxUSE_SOCKETS
24
25 #include "wx/socket.h"
26
27 #ifndef WX_PRECOMP
28 #include "wx/object.h"
29 #include "wx/string.h"
30 #include "wx/intl.h"
31 #include "wx/log.h"
32 #include "wx/event.h"
33 #include "wx/app.h"
34 #include "wx/utils.h"
35 #include "wx/timer.h"
36 #include "wx/module.h"
37 #endif
38
39 #include "wx/apptrait.h"
40 #include "wx/sckaddr.h"
41 #include "wx/stopwatch.h"
42 #include "wx/thread.h"
43 #include "wx/evtloop.h"
44
45 // DLL options compatibility check:
46 #include "wx/build.h"
47 WX_CHECK_BUILD_OPTIONS("wxNet")
48
49 // --------------------------------------------------------------------------
50 // macros and constants
51 // --------------------------------------------------------------------------
52
53 // discard buffer
54 #define MAX_DISCARD_SIZE (10 * 1024)
55
56 #define wxTRACE_Socket _T("wxSocket")
57
58 // --------------------------------------------------------------------------
59 // wxWin macros
60 // --------------------------------------------------------------------------
61
62 IMPLEMENT_CLASS(wxSocketBase, wxObject)
63 IMPLEMENT_CLASS(wxSocketServer, wxSocketBase)
64 IMPLEMENT_CLASS(wxSocketClient, wxSocketBase)
65 IMPLEMENT_CLASS(wxDatagramSocket, wxSocketBase)
66 IMPLEMENT_DYNAMIC_CLASS(wxSocketEvent, wxEvent)
67
68 // --------------------------------------------------------------------------
69 // private classes
70 // --------------------------------------------------------------------------
71
72 class wxSocketState : public wxObject
73 {
74 public:
75 wxSocketFlags m_flags;
76 wxSocketEventFlags m_eventmask;
77 bool m_notify;
78 void *m_clientData;
79
80 public:
81 wxSocketState() : wxObject() {}
82
83 DECLARE_NO_COPY_CLASS(wxSocketState)
84 };
85
86 // Conditionally make the socket non-blocking for the lifetime of this object.
87 class wxSocketUnblocker
88 {
89 public:
90 wxSocketUnblocker(GSocket *socket, bool unblock = true)
91 : m_socket(socket),
92 m_unblock(unblock)
93 {
94 if ( m_unblock )
95 m_socket->SetNonBlocking(true);
96 }
97
98 ~wxSocketUnblocker()
99 {
100 if ( m_unblock )
101 m_socket->SetNonBlocking(false);
102 }
103
104 private:
105 GSocket * const m_socket;
106 bool m_unblock;
107
108 DECLARE_NO_COPY_CLASS(wxSocketUnblocker)
109 };
110
111 // ============================================================================
112 // GSocketManager
113 // ============================================================================
114
115 GSocketManager *GSocketManager::ms_manager = NULL;
116
117 /* static */
118 void GSocketManager::Set(GSocketManager *manager)
119 {
120 wxASSERT_MSG( !ms_manager, "too late to set manager now" );
121
122 ms_manager = manager;
123 }
124
125 /* static */
126 void GSocketManager::Init()
127 {
128 wxASSERT_MSG( !ms_manager, "shouldn't be initialized twice" );
129
130 /*
131 Details: Initialize() creates a hidden window as a sink for socket
132 events, such as 'read completed'. wxMSW has only one message loop
133 for the main thread. If Initialize is called in a secondary thread,
134 the socket window will be created for the secondary thread, but
135 since there is no message loop on this thread, it will never
136 receive events and all socket operations will time out.
137 BTW, the main thread must not be stopped using sleep or block
138 on a semaphore (a bad idea in any case) or socket operations
139 will time out.
140
141 On the Mac side, Initialize() stores a pointer to the CFRunLoop for
142 the main thread. Because secondary threads do not have run loops,
143 adding event notifications to the "Current" loop would have no
144 effect at all, events would never fire.
145 */
146 wxASSERT_MSG( wxIsMainThread(),
147 "sockets must be initialized from the main thread" );
148
149 wxAppConsole * const app = wxAppConsole::GetInstance();
150 wxCHECK_RET( app, "sockets can't be initialized without wxApp" );
151
152 ms_manager = app->GetTraits()->GetSocketManager();
153 }
154
155 // ==========================================================================
156 // GSocketBase
157 // ==========================================================================
158
159 /* static */
160 GSocket *GSocketBase::Create()
161 {
162 GSocket * const newsocket = new GSocket();
163 if ( !GSocketManager::Get()->Init_Socket(newsocket) )
164 {
165 delete newsocket;
166 return NULL;
167 }
168
169 return newsocket;
170 }
171
172 GSocketBase::GSocketBase()
173 {
174 m_fd = INVALID_SOCKET;
175 m_detected = 0;
176 m_local = NULL;
177 m_peer = NULL;
178 m_error = GSOCK_NOERROR;
179 m_server = false;
180 m_stream = true;
181 m_non_blocking = false;
182 #ifdef __WINDOWS__
183 m_timeout.tv_sec = 10 * 60; /* 10 minutes */
184 m_timeout.tv_usec = 0;
185 #else
186 m_timeout = 10*60*1000; /* 10 minutes * 60 sec * 1000 millisec */
187 #endif
188
189 m_establishing = false;
190 m_reusable = false;
191 m_broadcast = false;
192 m_dobind = true;
193 m_initialRecvBufferSize = -1;
194 m_initialSendBufferSize = -1;
195
196 for ( int i = 0; i < GSOCK_MAX_EVENT; i++ )
197 m_cbacks[i] = NULL;
198 }
199
200 GSocketBase::~GSocketBase()
201 {
202 if (m_fd != INVALID_SOCKET)
203 Shutdown();
204
205 if (m_local)
206 GAddress_destroy(m_local);
207
208 if (m_peer)
209 GAddress_destroy(m_peer);
210
211 // cast is ok as all GSocketBase objects we have in our code are really
212 // GSockets
213 GSocketManager::Get()->Destroy_Socket(static_cast<GSocket *>(this));
214 }
215
216 /* GSocket_Shutdown:
217 * Disallow further read/write operations on this socket, close
218 * the fd and disable all callbacks.
219 */
220 void GSocketBase::Shutdown()
221 {
222 if ( m_fd != INVALID_SOCKET )
223 {
224 shutdown(m_fd, 1 /* SD_SEND */);
225 Close();
226 }
227
228 /* Disable GUI callbacks */
229 for ( int evt = 0; evt < GSOCK_MAX_EVENT; evt++ )
230 m_cbacks[evt] = NULL;
231
232 m_detected = GSOCK_LOST_FLAG;
233 }
234
235 // ==========================================================================
236 // wxSocketBase
237 // ==========================================================================
238
239 // --------------------------------------------------------------------------
240 // Initialization and shutdown
241 // --------------------------------------------------------------------------
242
243 // FIXME-MT: all this is MT-unsafe, of course, we should protect all accesses
244 // to m_countInit with a crit section
245 size_t wxSocketBase::m_countInit = 0;
246
247 bool wxSocketBase::IsInitialized()
248 {
249 return m_countInit > 0;
250 }
251
252 bool wxSocketBase::Initialize()
253 {
254 if ( !m_countInit++ )
255 {
256 if ( !GSocket_Init() )
257 {
258 m_countInit--;
259
260 return false;
261 }
262 }
263
264 return true;
265 }
266
267 void wxSocketBase::Shutdown()
268 {
269 // we should be initialized
270 wxASSERT_MSG( m_countInit > 0, _T("extra call to Shutdown()") );
271 if ( --m_countInit == 0 )
272 {
273 GSocket_Cleanup();
274 }
275 }
276
277 // --------------------------------------------------------------------------
278 // Ctor and dtor
279 // --------------------------------------------------------------------------
280
281 void wxSocketBase::Init()
282 {
283 m_socket = NULL;
284 m_type = wxSOCKET_UNINIT;
285
286 // state
287 m_flags = 0;
288 m_connected =
289 m_establishing =
290 m_reading =
291 m_writing =
292 m_error =
293 m_closed = false;
294 m_lcount = 0;
295 m_timeout = 600;
296 m_beingDeleted = false;
297
298 // pushback buffer
299 m_unread = NULL;
300 m_unrd_size = 0;
301 m_unrd_cur = 0;
302
303 // events
304 m_id = wxID_ANY;
305 m_handler = NULL;
306 m_clientData = NULL;
307 m_notify = false;
308 m_eventmask = 0;
309
310 if ( !IsInitialized() )
311 {
312 // this Initialize() will be undone by wxSocketModule::OnExit(), all
313 // the other calls to it should be matched by a call to Shutdown()
314 Initialize();
315 }
316 }
317
318 wxSocketBase::wxSocketBase()
319 {
320 Init();
321 }
322
323 wxSocketBase::wxSocketBase(wxSocketFlags flags, wxSocketType type)
324 {
325 Init();
326
327 SetFlags(flags);
328
329 m_type = type;
330 }
331
332 wxSocketBase::~wxSocketBase()
333 {
334 // Just in case the app called Destroy() *and* then deleted the socket
335 // immediately: don't leave dangling pointers.
336 wxAppTraits *traits = wxTheApp ? wxTheApp->GetTraits() : NULL;
337 if ( traits )
338 traits->RemoveFromPendingDelete(this);
339
340 // Shutdown and close the socket
341 if (!m_beingDeleted)
342 Close();
343
344 // Destroy the GSocket object
345 if (m_socket)
346 delete m_socket;
347
348 // Free the pushback buffer
349 if (m_unread)
350 free(m_unread);
351 }
352
353 bool wxSocketBase::Destroy()
354 {
355 // Delayed destruction: the socket will be deleted during the next idle
356 // loop iteration. This ensures that all pending events have been
357 // processed.
358 m_beingDeleted = true;
359
360 // Shutdown and close the socket
361 Close();
362
363 // Supress events from now on
364 Notify(false);
365
366 // schedule this object for deletion
367 wxAppTraits *traits = wxTheApp ? wxTheApp->GetTraits() : NULL;
368 if ( traits )
369 {
370 // let the traits object decide what to do with us
371 traits->ScheduleForDestroy(this);
372 }
373 else // no app or no traits
374 {
375 // in wxBase we might have no app object at all, don't leak memory
376 delete this;
377 }
378
379 return true;
380 }
381
382 // --------------------------------------------------------------------------
383 // Basic IO calls
384 // --------------------------------------------------------------------------
385
386 // The following IO operations update m_error and m_lcount:
387 // {Read, Write, ReadMsg, WriteMsg, Peek, Unread, Discard}
388 //
389 // TODO: Should Connect, Accept and AcceptWith update m_error?
390
391 bool wxSocketBase::Close()
392 {
393 // Interrupt pending waits
394 InterruptWait();
395
396 if (m_socket)
397 {
398 // Disable callbacks
399 m_socket->UnsetCallback(
400 GSOCK_INPUT_FLAG |
401 GSOCK_OUTPUT_FLAG |
402 GSOCK_LOST_FLAG |
403 GSOCK_CONNECTION_FLAG
404 );
405
406 // Shutdown the connection
407 m_socket->Shutdown();
408 }
409
410 m_connected = false;
411 m_establishing = false;
412 return true;
413 }
414
415 wxSocketBase& wxSocketBase::Read(void* buffer, wxUint32 nbytes)
416 {
417 // Mask read events
418 m_reading = true;
419
420 m_lcount = DoRead(buffer, nbytes);
421
422 // If in wxSOCKET_WAITALL mode, all bytes should have been read.
423 if (m_flags & wxSOCKET_WAITALL)
424 m_error = (m_lcount != nbytes);
425 else
426 m_error = (m_lcount == 0);
427
428 // Allow read events from now on
429 m_reading = false;
430
431 return *this;
432 }
433
434 wxUint32 wxSocketBase::DoRead(void* buffer_, wxUint32 nbytes)
435 {
436 // We use pointer arithmetic here which doesn't work with void pointers.
437 char *buffer = static_cast<char *>(buffer_);
438
439 // Try the push back buffer first, even before checking whether the socket
440 // is valid to allow reading previously pushed back data from an already
441 // closed socket.
442 wxUint32 total = GetPushback(buffer, nbytes, false);
443 nbytes -= total;
444 buffer += total;
445
446 // If it's indeed closed or if read everything, there is nothing more to do.
447 if ( !m_socket || !nbytes )
448 return total;
449
450 wxCHECK_MSG( buffer, 0, "NULL buffer" );
451
452
453 // wxSOCKET_NOWAIT overrides all the other flags and means that we are
454 // polling the socket and don't block at all.
455 if ( m_flags & wxSOCKET_NOWAIT )
456 {
457 wxSocketUnblocker unblock(m_socket);
458 int ret = m_socket->Read(buffer, nbytes);
459 if ( ret < 0 )
460 return 0;
461
462 total += ret;
463 }
464 else // blocking socket
465 {
466 for ( ;; )
467 {
468 // Wait until socket becomes ready for reading dispatching the GUI
469 // events in the meanwhile unless wxSOCKET_BLOCK was explicitly
470 // specified to disable this.
471 if ( !(m_flags & wxSOCKET_BLOCK) && !WaitForRead() )
472 break;
473
474 const int ret = m_socket->Read(buffer, nbytes);
475 if ( ret == 0 )
476 {
477 // for connection-oriented (e.g. TCP) sockets we can only read
478 // 0 bytes if the other end has been closed, and for
479 // connectionless ones (UDP) this flag doesn't make sense
480 // anyhow so we can set it to true too without doing any harm
481 m_closed = true;
482 break;
483 }
484
485 if ( ret < 0 )
486 {
487 // this will be always interpreted as error by Read()
488 return 0;
489 }
490
491 total += ret;
492
493 // If wxSOCKET_WAITALL is not set, we can leave now as we did read
494 // something and we don't need to wait for all nbytes bytes to be
495 // read.
496 if ( !(m_flags & wxSOCKET_WAITALL) )
497 break;
498
499 // Otherwise continue reading until we do read everything.
500 nbytes -= ret;
501 if ( !nbytes )
502 break;
503
504 buffer += ret;
505 }
506 }
507
508 return total;
509 }
510
511 wxSocketBase& wxSocketBase::ReadMsg(void* buffer, wxUint32 nbytes)
512 {
513 wxUint32 len, len2, sig, total;
514 bool error;
515 int old_flags;
516 struct
517 {
518 unsigned char sig[4];
519 unsigned char len[4];
520 } msg;
521
522 // Mask read events
523 m_reading = true;
524
525 total = 0;
526 error = true;
527 old_flags = m_flags;
528 SetFlags((m_flags & wxSOCKET_BLOCK) | wxSOCKET_WAITALL);
529
530 if (DoRead(&msg, sizeof(msg)) != sizeof(msg))
531 goto exit;
532
533 sig = (wxUint32)msg.sig[0];
534 sig |= (wxUint32)(msg.sig[1] << 8);
535 sig |= (wxUint32)(msg.sig[2] << 16);
536 sig |= (wxUint32)(msg.sig[3] << 24);
537
538 if (sig != 0xfeeddead)
539 {
540 wxLogWarning(_("wxSocket: invalid signature in ReadMsg."));
541 goto exit;
542 }
543
544 len = (wxUint32)msg.len[0];
545 len |= (wxUint32)(msg.len[1] << 8);
546 len |= (wxUint32)(msg.len[2] << 16);
547 len |= (wxUint32)(msg.len[3] << 24);
548
549 if (len > nbytes)
550 {
551 len2 = len - nbytes;
552 len = nbytes;
553 }
554 else
555 len2 = 0;
556
557 // Don't attempt to read if the msg was zero bytes long.
558 if (len)
559 {
560 total = DoRead(buffer, len);
561
562 if (total != len)
563 goto exit;
564 }
565
566 if (len2)
567 {
568 char *discard_buffer = new char[MAX_DISCARD_SIZE];
569 long discard_len;
570
571 // NOTE: discarded bytes don't add to m_lcount.
572 do
573 {
574 discard_len = ((len2 > MAX_DISCARD_SIZE)? MAX_DISCARD_SIZE : len2);
575 discard_len = DoRead(discard_buffer, (wxUint32)discard_len);
576 len2 -= (wxUint32)discard_len;
577 }
578 while ((discard_len > 0) && len2);
579
580 delete [] discard_buffer;
581
582 if (len2 != 0)
583 goto exit;
584 }
585 if (DoRead(&msg, sizeof(msg)) != sizeof(msg))
586 goto exit;
587
588 sig = (wxUint32)msg.sig[0];
589 sig |= (wxUint32)(msg.sig[1] << 8);
590 sig |= (wxUint32)(msg.sig[2] << 16);
591 sig |= (wxUint32)(msg.sig[3] << 24);
592
593 if (sig != 0xdeadfeed)
594 {
595 wxLogWarning(_("wxSocket: invalid signature in ReadMsg."));
596 goto exit;
597 }
598
599 // everything was OK
600 error = false;
601
602 exit:
603 m_error = error;
604 m_lcount = total;
605 m_reading = false;
606 SetFlags(old_flags);
607
608 return *this;
609 }
610
611 wxSocketBase& wxSocketBase::Peek(void* buffer, wxUint32 nbytes)
612 {
613 // Mask read events
614 m_reading = true;
615
616 m_lcount = DoRead(buffer, nbytes);
617 Pushback(buffer, m_lcount);
618
619 // If in wxSOCKET_WAITALL mode, all bytes should have been read.
620 if (m_flags & wxSOCKET_WAITALL)
621 m_error = (m_lcount != nbytes);
622 else
623 m_error = (m_lcount == 0);
624
625 // Allow read events again
626 m_reading = false;
627
628 return *this;
629 }
630
631 wxSocketBase& wxSocketBase::Write(const void *buffer, wxUint32 nbytes)
632 {
633 // Mask write events
634 m_writing = true;
635
636 m_lcount = DoWrite(buffer, nbytes);
637
638 // If in wxSOCKET_WAITALL mode, all bytes should have been written.
639 if (m_flags & wxSOCKET_WAITALL)
640 m_error = (m_lcount != nbytes);
641 else
642 m_error = (m_lcount == 0);
643
644 // Allow write events again
645 m_writing = false;
646
647 return *this;
648 }
649
650 // This function is a mirror image of DoRead() except that it doesn't use the
651 // push back buffer, please see comments there
652 wxUint32 wxSocketBase::DoWrite(const void *buffer_, wxUint32 nbytes)
653 {
654 const char *buffer = static_cast<const char *>(buffer_);
655
656 // Return if there is nothing to read or the socket is (already?) closed.
657 if ( !m_socket || !nbytes )
658 return 0;
659
660 wxCHECK_MSG( buffer, 0, "NULL buffer" );
661
662 wxUint32 total = 0;
663 if ( m_flags & wxSOCKET_NOWAIT )
664 {
665 wxSocketUnblocker unblock(m_socket);
666 const int ret = m_socket->Write(buffer, nbytes);
667 if ( ret > 0 )
668 total += ret;
669 }
670 else // blocking socket
671 {
672 for ( ;; )
673 {
674 if ( !(m_flags & wxSOCKET_BLOCK) && !WaitForWrite() )
675 break;
676
677 const int ret = m_socket->Write(buffer, nbytes);
678 if ( ret == 0 )
679 {
680 m_closed = true;
681 break;
682 }
683
684 if ( ret < 0 )
685 return 0;
686
687 total += ret;
688 if ( !(m_flags & wxSOCKET_WAITALL) )
689 break;
690
691 nbytes -= ret;
692 if ( !nbytes )
693 break;
694
695 buffer += ret;
696 }
697 }
698
699 return total;
700 }
701
702 wxSocketBase& wxSocketBase::WriteMsg(const void *buffer, wxUint32 nbytes)
703 {
704 wxUint32 total;
705 bool error;
706 struct
707 {
708 unsigned char sig[4];
709 unsigned char len[4];
710 } msg;
711
712 // Mask write events
713 m_writing = true;
714
715 error = true;
716 total = 0;
717 SetFlags((m_flags & wxSOCKET_BLOCK) | wxSOCKET_WAITALL);
718
719 msg.sig[0] = (unsigned char) 0xad;
720 msg.sig[1] = (unsigned char) 0xde;
721 msg.sig[2] = (unsigned char) 0xed;
722 msg.sig[3] = (unsigned char) 0xfe;
723
724 msg.len[0] = (unsigned char) (nbytes & 0xff);
725 msg.len[1] = (unsigned char) ((nbytes >> 8) & 0xff);
726 msg.len[2] = (unsigned char) ((nbytes >> 16) & 0xff);
727 msg.len[3] = (unsigned char) ((nbytes >> 24) & 0xff);
728
729 if (DoWrite(&msg, sizeof(msg)) < sizeof(msg))
730 goto exit;
731
732 total = DoWrite(buffer, nbytes);
733
734 if (total < nbytes)
735 goto exit;
736
737 msg.sig[0] = (unsigned char) 0xed;
738 msg.sig[1] = (unsigned char) 0xfe;
739 msg.sig[2] = (unsigned char) 0xad;
740 msg.sig[3] = (unsigned char) 0xde;
741 msg.len[0] =
742 msg.len[1] =
743 msg.len[2] =
744 msg.len[3] = (char) 0;
745
746 if ((DoWrite(&msg, sizeof(msg))) < sizeof(msg))
747 goto exit;
748
749 // everything was OK
750 error = false;
751
752 exit:
753 m_error = error;
754 m_lcount = total;
755 m_writing = false;
756
757 return *this;
758 }
759
760 wxSocketBase& wxSocketBase::Unread(const void *buffer, wxUint32 nbytes)
761 {
762 if (nbytes != 0)
763 Pushback(buffer, nbytes);
764
765 m_error = false;
766 m_lcount = nbytes;
767
768 return *this;
769 }
770
771 wxSocketBase& wxSocketBase::Discard()
772 {
773 char *buffer = new char[MAX_DISCARD_SIZE];
774 wxUint32 ret;
775 wxUint32 total = 0;
776
777 // Mask read events
778 m_reading = true;
779
780 SetFlags(wxSOCKET_NOWAIT);
781
782 do
783 {
784 ret = DoRead(buffer, MAX_DISCARD_SIZE);
785 total += ret;
786 }
787 while (ret == MAX_DISCARD_SIZE);
788
789 delete[] buffer;
790 m_lcount = total;
791 m_error = false;
792
793 // Allow read events again
794 m_reading = false;
795
796 return *this;
797 }
798
799 // --------------------------------------------------------------------------
800 // Wait functions
801 // --------------------------------------------------------------------------
802
803 /* GSocket_Select:
804 * Polls the socket to determine its status. This function will
805 * check for the events specified in the 'flags' parameter, and
806 * it will return a mask indicating which operations can be
807 * performed. This function won't block, regardless of the
808 * mode (blocking | nonblocking) of the socket.
809 */
810 GSocketEventFlags GSocketBase::Select(GSocketEventFlags flags)
811 {
812 assert(this);
813
814 GSocketEventFlags result = 0;
815 fd_set readfds;
816 fd_set writefds;
817 fd_set exceptfds;
818 struct timeval tv;
819
820 if (m_fd == -1)
821 return (GSOCK_LOST_FLAG & flags);
822
823 /* Do not use a static struct, Linux can garble it */
824 tv.tv_sec = 0;
825 tv.tv_usec = 0;
826
827 wxFD_ZERO(&readfds);
828 wxFD_ZERO(&writefds);
829 wxFD_ZERO(&exceptfds);
830 wxFD_SET(m_fd, &readfds);
831 if (flags & GSOCK_OUTPUT_FLAG || flags & GSOCK_CONNECTION_FLAG)
832 wxFD_SET(m_fd, &writefds);
833 wxFD_SET(m_fd, &exceptfds);
834
835 /* Check 'sticky' CONNECTION flag first */
836 result |= GSOCK_CONNECTION_FLAG & m_detected;
837
838 /* If we have already detected a LOST event, then don't try
839 * to do any further processing.
840 */
841 if ((m_detected & GSOCK_LOST_FLAG) != 0)
842 {
843 m_establishing = false;
844 return (GSOCK_LOST_FLAG & flags);
845 }
846
847 /* Try select now */
848 if (select(m_fd + 1, &readfds, &writefds, &exceptfds, &tv) < 0)
849 {
850 /* What to do here? */
851 return (result & flags);
852 }
853
854 /* Check for exceptions and errors */
855 if (wxFD_ISSET(m_fd, &exceptfds))
856 {
857 m_establishing = false;
858 m_detected = GSOCK_LOST_FLAG;
859
860 /* LOST event: Abort any further processing */
861 return (GSOCK_LOST_FLAG & flags);
862 }
863
864 /* Check for readability */
865 if (wxFD_ISSET(m_fd, &readfds))
866 {
867 result |= GSOCK_INPUT_FLAG;
868
869 if (m_server && m_stream)
870 {
871 /* This is a TCP server socket that detected a connection.
872 While the INPUT_FLAG is also set, it doesn't matter on
873 this kind of sockets, as we can only Accept() from them. */
874 m_detected |= GSOCK_CONNECTION_FLAG;
875 }
876 }
877
878 /* Check for writability */
879 if (wxFD_ISSET(m_fd, &writefds))
880 {
881 if (m_establishing && !m_server)
882 {
883 int error;
884 SOCKOPTLEN_T len = sizeof(error);
885 m_establishing = false;
886 getsockopt(m_fd, SOL_SOCKET, SO_ERROR, (char*)&error, &len);
887
888 if (error)
889 {
890 m_detected = GSOCK_LOST_FLAG;
891
892 /* LOST event: Abort any further processing */
893 return (GSOCK_LOST_FLAG & flags);
894 }
895 else
896 {
897 m_detected |= GSOCK_CONNECTION_FLAG;
898 }
899 }
900 else
901 {
902 result |= GSOCK_OUTPUT_FLAG;
903 }
904 }
905
906 return (result | m_detected) & flags;
907 }
908
909 // All Wait functions poll the socket using GSocket_Select() to
910 // check for the specified combination of conditions, until one
911 // of these conditions become true, an error occurs, or the
912 // timeout elapses. The polling loop runs the event loop so that
913 // this won't block the GUI.
914
915 bool
916 wxSocketBase::DoWait(long seconds, long milliseconds, wxSocketEventFlags flags)
917 {
918 wxCHECK_MSG( m_socket, false, "can't wait on invalid socket" );
919
920 // This can be set to true from Interrupt() to exit this function a.s.a.p.
921 m_interrupt = false;
922
923
924 // Use either the provided timeout or the default timeout value associated
925 // with this socket.
926 //
927 // TODO: allow waiting forever, see #9443
928 const long timeout = seconds == -1 ? m_timeout * 1000
929 : seconds * 1000 + milliseconds;
930 const wxMilliClock_t timeEnd = wxGetLocalTimeMillis() + timeout;
931
932 // Get the active event loop which we'll use for the message dispatching
933 // when running in the main thread
934 wxEventLoopBase *eventLoop;
935 if ( wxIsMainThread() )
936 {
937 eventLoop = wxEventLoop::GetActive();
938 }
939 else // in worker thread
940 {
941 // We never dispatch messages from threads other than the main one.
942 eventLoop = NULL;
943 }
944
945 // Wait in an active polling loop: notice that the loop is executed at
946 // least once, even if timeout is 0 (i.e. polling).
947 bool gotEvent = false;
948 for ( ;; )
949 {
950 // We always stop waiting when the connection is lost as it doesn't
951 // make sense to continue further, even if GSOCK_LOST_FLAG is not
952 // specified in flags to wait for.
953 const GSocketEventFlags
954 result = m_socket->Select(flags | GSOCK_LOST_FLAG);
955
956 // Incoming connection (server) or connection established (client)?
957 if ( result & GSOCK_CONNECTION_FLAG )
958 {
959 m_connected = true;
960 m_establishing = false;
961 gotEvent = true;
962 break;
963 }
964
965 // Data available or output buffer ready?
966 if ( (result & GSOCK_INPUT_FLAG) || (result & GSOCK_OUTPUT_FLAG) )
967 {
968 gotEvent = true;
969 break;
970 }
971
972 // Connection lost
973 if ( result & GSOCK_LOST_FLAG )
974 {
975 m_connected = false;
976 m_establishing = false;
977 if ( flags & GSOCK_LOST_FLAG )
978 gotEvent = true;
979 break;
980 }
981
982 if ( m_interrupt )
983 break;
984
985 // Wait more?
986 const wxMilliClock_t timeNow = wxGetLocalTimeMillis();
987 if ( timeNow >= timeEnd )
988 break;
989
990 if ( eventLoop )
991 {
992 // This function is only called if wxSOCKET_BLOCK flag was not used
993 // and so we should dispatch the events if there is an event loop
994 // capable of doing it.
995 if ( eventLoop->Pending() )
996 eventLoop->Dispatch();
997 }
998 #if wxUSE_THREADS
999 else // no event loop or waiting in another thread
1000 {
1001 // We're busy waiting but at least give up the rest of our current
1002 // time slice.
1003 wxThread::Yield();
1004 }
1005 #endif // wxUSE_THREADS
1006 }
1007
1008 return gotEvent;
1009 }
1010
1011 bool wxSocketBase::Wait(long seconds, long milliseconds)
1012 {
1013 return DoWait(seconds, milliseconds,
1014 GSOCK_INPUT_FLAG |
1015 GSOCK_OUTPUT_FLAG |
1016 GSOCK_CONNECTION_FLAG |
1017 GSOCK_LOST_FLAG
1018 );
1019 }
1020
1021 bool wxSocketBase::WaitForRead(long seconds, long milliseconds)
1022 {
1023 // Check pushback buffer before entering DoWait
1024 if ( m_unread )
1025 return true;
1026
1027 // Note that GSOCK_INPUT_LOST has to be explicitly passed to DoWait
1028 // because of the semantics of WaitForRead: a return value of true means
1029 // that a GSocket_Read call will return immediately, not that there is
1030 // actually data to read.
1031 return DoWait(seconds, milliseconds, GSOCK_INPUT_FLAG | GSOCK_LOST_FLAG);
1032 }
1033
1034
1035 bool wxSocketBase::WaitForWrite(long seconds, long milliseconds)
1036 {
1037 return DoWait(seconds, milliseconds, GSOCK_OUTPUT_FLAG | GSOCK_LOST_FLAG);
1038 }
1039
1040 bool wxSocketBase::WaitForLost(long seconds, long milliseconds)
1041 {
1042 return DoWait(seconds, milliseconds, GSOCK_LOST_FLAG);
1043 }
1044
1045 // --------------------------------------------------------------------------
1046 // Miscellaneous
1047 // --------------------------------------------------------------------------
1048
1049 //
1050 // Get local or peer address
1051 //
1052
1053 bool wxSocketBase::GetPeer(wxSockAddress& addr_man) const
1054 {
1055 GAddress *peer;
1056
1057 if (!m_socket)
1058 return false;
1059
1060 peer = m_socket->GetPeer();
1061
1062 // copying a null address would just trigger an assert anyway
1063
1064 if (!peer)
1065 return false;
1066
1067 addr_man.SetAddress(peer);
1068 GAddress_destroy(peer);
1069
1070 return true;
1071 }
1072
1073 bool wxSocketBase::GetLocal(wxSockAddress& addr_man) const
1074 {
1075 GAddress *local;
1076
1077 if (!m_socket)
1078 return false;
1079
1080 local = m_socket->GetLocal();
1081 addr_man.SetAddress(local);
1082 GAddress_destroy(local);
1083
1084 return true;
1085 }
1086
1087 //
1088 // Save and restore socket state
1089 //
1090
1091 void wxSocketBase::SaveState()
1092 {
1093 wxSocketState *state;
1094
1095 state = new wxSocketState();
1096
1097 state->m_flags = m_flags;
1098 state->m_notify = m_notify;
1099 state->m_eventmask = m_eventmask;
1100 state->m_clientData = m_clientData;
1101
1102 m_states.Append(state);
1103 }
1104
1105 void wxSocketBase::RestoreState()
1106 {
1107 wxList::compatibility_iterator node;
1108 wxSocketState *state;
1109
1110 node = m_states.GetLast();
1111 if (!node)
1112 return;
1113
1114 state = (wxSocketState *)node->GetData();
1115
1116 m_flags = state->m_flags;
1117 m_notify = state->m_notify;
1118 m_eventmask = state->m_eventmask;
1119 m_clientData = state->m_clientData;
1120
1121 m_states.Erase(node);
1122 delete state;
1123 }
1124
1125 //
1126 // Timeout and flags
1127 //
1128
1129 void wxSocketBase::SetTimeout(long seconds)
1130 {
1131 m_timeout = seconds;
1132
1133 if (m_socket)
1134 m_socket->SetTimeout(m_timeout * 1000);
1135 }
1136
1137 void wxSocketBase::SetFlags(wxSocketFlags flags)
1138 {
1139 // Do some sanity checking on the flags used: not all values can be used
1140 // together.
1141 wxASSERT_MSG( !(flags & wxSOCKET_NOWAIT) ||
1142 !(flags & (wxSOCKET_WAITALL | wxSOCKET_BLOCK)),
1143 "Using wxSOCKET_WAITALL or wxSOCKET_BLOCK with "
1144 "wxSOCKET_NOWAIT doesn't make sense" );
1145
1146 m_flags = flags;
1147 }
1148
1149
1150 // --------------------------------------------------------------------------
1151 // Event handling
1152 // --------------------------------------------------------------------------
1153
1154 // A note on how events are processed, which is probably the most
1155 // difficult thing to get working right while keeping the same API
1156 // and functionality for all platforms.
1157 //
1158 // When GSocket detects an event, it calls wx_socket_callback, which in
1159 // turn just calls wxSocketBase::OnRequest in the corresponding wxSocket
1160 // object. OnRequest does some housekeeping, and if the event is to be
1161 // propagated to the user, it creates a new wxSocketEvent object and
1162 // posts it. The event is not processed immediately, but delayed with
1163 // AddPendingEvent instead. This is necessary in order to decouple the
1164 // event processing from wx_socket_callback; otherwise, subsequent IO
1165 // calls made from the user event handler would fail, as gtk callbacks
1166 // are not reentrant.
1167 //
1168 // Note that, unlike events, user callbacks (now deprecated) are _not_
1169 // decoupled from wx_socket_callback and thus they suffer from a variety
1170 // of problems. Avoid them where possible and use events instead.
1171
1172 extern "C"
1173 void LINKAGEMODE wx_socket_callback(GSocket * WXUNUSED(socket),
1174 GSocketEvent notification,
1175 char *cdata)
1176 {
1177 wxSocketBase *sckobj = (wxSocketBase *)cdata;
1178
1179 sckobj->OnRequest((wxSocketNotify) notification);
1180 }
1181
1182 void wxSocketBase::OnRequest(wxSocketNotify notification)
1183 {
1184 switch(notification)
1185 {
1186 case wxSOCKET_CONNECTION:
1187 m_establishing = false;
1188 m_connected = true;
1189 break;
1190
1191 // If we are in the middle of a R/W operation, do not
1192 // propagate events to users. Also, filter 'late' events
1193 // which are no longer valid.
1194
1195 case wxSOCKET_INPUT:
1196 if (m_reading || !m_socket->Select(GSOCK_INPUT_FLAG))
1197 return;
1198 break;
1199
1200 case wxSOCKET_OUTPUT:
1201 if (m_writing || !m_socket->Select(GSOCK_OUTPUT_FLAG))
1202 return;
1203 break;
1204
1205 case wxSOCKET_LOST:
1206 m_connected = false;
1207 m_establishing = false;
1208 break;
1209 }
1210
1211 // Schedule the event
1212
1213 wxSocketEventFlags flag = 0;
1214 wxUnusedVar(flag);
1215 switch (notification)
1216 {
1217 case GSOCK_INPUT: flag = GSOCK_INPUT_FLAG; break;
1218 case GSOCK_OUTPUT: flag = GSOCK_OUTPUT_FLAG; break;
1219 case GSOCK_CONNECTION: flag = GSOCK_CONNECTION_FLAG; break;
1220 case GSOCK_LOST: flag = GSOCK_LOST_FLAG; break;
1221 default:
1222 wxLogWarning(_("wxSocket: unknown event!."));
1223 return;
1224 }
1225
1226 if (((m_eventmask & flag) == flag) && m_notify)
1227 {
1228 if (m_handler)
1229 {
1230 wxSocketEvent event(m_id);
1231 event.m_event = notification;
1232 event.m_clientData = m_clientData;
1233 event.SetEventObject(this);
1234
1235 m_handler->AddPendingEvent(event);
1236 }
1237 }
1238 }
1239
1240 void wxSocketBase::Notify(bool notify)
1241 {
1242 m_notify = notify;
1243 if (m_socket)
1244 m_socket->Notify(notify);
1245 }
1246
1247 void wxSocketBase::SetNotify(wxSocketEventFlags flags)
1248 {
1249 m_eventmask = flags;
1250 }
1251
1252 void wxSocketBase::SetEventHandler(wxEvtHandler& handler, int id)
1253 {
1254 m_handler = &handler;
1255 m_id = id;
1256 }
1257
1258 // --------------------------------------------------------------------------
1259 // Pushback buffer
1260 // --------------------------------------------------------------------------
1261
1262 void wxSocketBase::Pushback(const void *buffer, wxUint32 size)
1263 {
1264 if (!size) return;
1265
1266 if (m_unread == NULL)
1267 m_unread = malloc(size);
1268 else
1269 {
1270 void *tmp;
1271
1272 tmp = malloc(m_unrd_size + size);
1273 memcpy((char *)tmp + size, m_unread, m_unrd_size);
1274 free(m_unread);
1275
1276 m_unread = tmp;
1277 }
1278
1279 m_unrd_size += size;
1280
1281 memcpy(m_unread, buffer, size);
1282 }
1283
1284 wxUint32 wxSocketBase::GetPushback(void *buffer, wxUint32 size, bool peek)
1285 {
1286 wxCHECK_MSG( buffer, 0, "NULL buffer" );
1287
1288 if (!m_unrd_size)
1289 return 0;
1290
1291 if (size > (m_unrd_size-m_unrd_cur))
1292 size = m_unrd_size-m_unrd_cur;
1293
1294 memcpy(buffer, (char *)m_unread + m_unrd_cur, size);
1295
1296 if (!peek)
1297 {
1298 m_unrd_cur += size;
1299 if (m_unrd_size == m_unrd_cur)
1300 {
1301 free(m_unread);
1302 m_unread = NULL;
1303 m_unrd_size = 0;
1304 m_unrd_cur = 0;
1305 }
1306 }
1307
1308 return size;
1309 }
1310
1311
1312 // ==========================================================================
1313 // wxSocketServer
1314 // ==========================================================================
1315
1316 // --------------------------------------------------------------------------
1317 // Ctor
1318 // --------------------------------------------------------------------------
1319
1320 wxSocketServer::wxSocketServer(const wxSockAddress& addr_man,
1321 wxSocketFlags flags)
1322 : wxSocketBase(flags, wxSOCKET_SERVER)
1323 {
1324 wxLogTrace( wxTRACE_Socket, _T("Opening wxSocketServer") );
1325
1326 m_socket = GSocket::Create();
1327
1328 if (!m_socket)
1329 {
1330 wxLogTrace( wxTRACE_Socket, _T("*** GSocket_new failed") );
1331 return;
1332 }
1333
1334 // Setup the socket as server
1335 m_socket->Notify(m_notify);
1336 m_socket->SetLocal(addr_man.GetAddress());
1337
1338 if (GetFlags() & wxSOCKET_REUSEADDR) {
1339 m_socket->SetReusable();
1340 }
1341 if (GetFlags() & wxSOCKET_BROADCAST) {
1342 m_socket->SetBroadcast();
1343 }
1344 if (GetFlags() & wxSOCKET_NOBIND) {
1345 m_socket->DontDoBind();
1346 }
1347
1348 if (m_socket->SetServer() != GSOCK_NOERROR)
1349 {
1350 delete m_socket;
1351 m_socket = NULL;
1352
1353 wxLogTrace( wxTRACE_Socket, _T("*** GSocket_SetServer failed") );
1354 return;
1355 }
1356
1357 m_socket->SetTimeout(m_timeout * 1000);
1358 m_socket->SetCallback(GSOCK_INPUT_FLAG | GSOCK_OUTPUT_FLAG |
1359 GSOCK_LOST_FLAG | GSOCK_CONNECTION_FLAG,
1360 wx_socket_callback, (char *)this);
1361
1362 wxLogTrace( wxTRACE_Socket, _T("wxSocketServer on fd %d"), m_socket->m_fd );
1363 }
1364
1365 // --------------------------------------------------------------------------
1366 // Accept
1367 // --------------------------------------------------------------------------
1368
1369 bool wxSocketServer::AcceptWith(wxSocketBase& sock, bool wait)
1370 {
1371 if (!m_socket)
1372 return false;
1373
1374 // If wait == false, then the call should be nonblocking.
1375 // When we are finished, we put the socket to blocking mode
1376 // again.
1377 wxSocketUnblocker unblock(m_socket, !wait);
1378 GSocket * const child_socket = m_socket->WaitConnection();
1379
1380 if (!child_socket)
1381 return false;
1382
1383 sock.m_type = wxSOCKET_BASE;
1384 sock.m_socket = child_socket;
1385 sock.m_connected = true;
1386
1387 sock.m_socket->SetTimeout(sock.m_timeout * 1000);
1388 sock.m_socket->SetCallback(GSOCK_INPUT_FLAG | GSOCK_OUTPUT_FLAG |
1389 GSOCK_LOST_FLAG | GSOCK_CONNECTION_FLAG,
1390 wx_socket_callback, (char *)&sock);
1391
1392 return true;
1393 }
1394
1395 wxSocketBase *wxSocketServer::Accept(bool wait)
1396 {
1397 wxSocketBase* sock = new wxSocketBase();
1398
1399 sock->SetFlags(m_flags);
1400
1401 if (!AcceptWith(*sock, wait))
1402 {
1403 sock->Destroy();
1404 sock = NULL;
1405 }
1406
1407 return sock;
1408 }
1409
1410 bool wxSocketServer::WaitForAccept(long seconds, long milliseconds)
1411 {
1412 return DoWait(seconds, milliseconds, GSOCK_CONNECTION_FLAG);
1413 }
1414
1415 bool wxSocketBase::GetOption(int level, int optname, void *optval, int *optlen)
1416 {
1417 wxASSERT_MSG( m_socket, _T("Socket not initialised") );
1418
1419 if (m_socket->GetSockOpt(level, optname, optval, optlen)
1420 != GSOCK_NOERROR)
1421 {
1422 return false;
1423 }
1424 return true;
1425 }
1426
1427 bool wxSocketBase::SetOption(int level, int optname, const void *optval,
1428 int optlen)
1429 {
1430 wxASSERT_MSG( m_socket, _T("Socket not initialised") );
1431
1432 if (m_socket->SetSockOpt(level, optname, optval, optlen)
1433 != GSOCK_NOERROR)
1434 {
1435 return false;
1436 }
1437 return true;
1438 }
1439
1440 bool wxSocketBase::SetLocal(const wxIPV4address& local)
1441 {
1442 GAddress* la = local.GetAddress();
1443
1444 // If the address is valid, save it for use when we call Connect
1445 if (la && la->m_addr)
1446 {
1447 m_localAddress = local;
1448
1449 return true;
1450 }
1451
1452 return false;
1453 }
1454
1455 // ==========================================================================
1456 // wxSocketClient
1457 // ==========================================================================
1458
1459 // --------------------------------------------------------------------------
1460 // Ctor and dtor
1461 // --------------------------------------------------------------------------
1462
1463 wxSocketClient::wxSocketClient(wxSocketFlags flags)
1464 : wxSocketBase(flags, wxSOCKET_CLIENT)
1465 {
1466 m_initialRecvBufferSize =
1467 m_initialSendBufferSize = -1;
1468 }
1469
1470 wxSocketClient::~wxSocketClient()
1471 {
1472 }
1473
1474 // --------------------------------------------------------------------------
1475 // Connect
1476 // --------------------------------------------------------------------------
1477
1478 bool wxSocketClient::DoConnect(const wxSockAddress& addr_man,
1479 const wxSockAddress* local,
1480 bool wait)
1481 {
1482 if (m_socket)
1483 {
1484 // Shutdown and destroy the socket
1485 Close();
1486 delete m_socket;
1487 }
1488
1489 m_socket = GSocket::Create();
1490 m_connected = false;
1491 m_establishing = false;
1492
1493 if (!m_socket)
1494 return false;
1495
1496 m_socket->SetTimeout(m_timeout * 1000);
1497 m_socket->SetCallback(
1498 GSOCK_INPUT_FLAG |
1499 GSOCK_OUTPUT_FLAG |
1500 GSOCK_LOST_FLAG |
1501 GSOCK_CONNECTION_FLAG,
1502 wx_socket_callback,
1503 (char *)this
1504 );
1505
1506 // If wait == false, then the call should be nonblocking. When we are
1507 // finished, we put the socket to blocking mode again.
1508 wxSocketUnblocker unblock(m_socket, !wait);
1509
1510 // Reuse makes sense for clients too, if we are trying to rebind to the same port
1511 if (GetFlags() & wxSOCKET_REUSEADDR)
1512 {
1513 m_socket->SetReusable();
1514 }
1515 if (GetFlags() & wxSOCKET_BROADCAST)
1516 {
1517 m_socket->SetBroadcast();
1518 }
1519 if (GetFlags() & wxSOCKET_NOBIND)
1520 {
1521 m_socket->DontDoBind();
1522 }
1523
1524 // If no local address was passed and one has been set, use the one that was Set
1525 if (!local && m_localAddress.GetAddress())
1526 {
1527 local = &m_localAddress;
1528 }
1529
1530 // Bind to the local IP address and port, when provided
1531 if (local)
1532 {
1533 GAddress* la = local->GetAddress();
1534
1535 if (la && la->m_addr)
1536 m_socket->SetLocal(la);
1537 }
1538
1539 #if defined(__WXMSW__) || defined(__WXGTK__)
1540 m_socket->SetInitialSocketBuffers(m_initialRecvBufferSize, m_initialSendBufferSize);
1541 #endif
1542
1543 m_socket->SetPeer(addr_man.GetAddress());
1544 const GSocketError err = m_socket->Connect(GSOCK_STREAMED);
1545
1546 //this will register for callbacks - must be called after m_socket->m_fd was initialized
1547 m_socket->Notify(m_notify);
1548
1549 if (err != GSOCK_NOERROR)
1550 {
1551 if (err == GSOCK_WOULDBLOCK)
1552 m_establishing = true;
1553
1554 return false;
1555 }
1556
1557 m_connected = true;
1558 return true;
1559 }
1560
1561 bool wxSocketClient::Connect(const wxSockAddress& addr_man, bool wait)
1562 {
1563 return DoConnect(addr_man, NULL, wait);
1564 }
1565
1566 bool wxSocketClient::Connect(const wxSockAddress& addr_man,
1567 const wxSockAddress& local,
1568 bool wait)
1569 {
1570 return DoConnect(addr_man, &local, wait);
1571 }
1572
1573 bool wxSocketClient::WaitOnConnect(long seconds, long milliseconds)
1574 {
1575 if ( m_connected )
1576 {
1577 // this happens if the initial attempt to connect succeeded without
1578 // blocking
1579 return true;
1580 }
1581
1582 wxCHECK_MSG( m_establishing && m_socket, false,
1583 "No connection establishment attempt in progress" );
1584
1585 // we must specify GSOCK_LOST_FLAG here explicitly because we must return
1586 // true if the connection establishment process is finished, whether it is
1587 // over because we successfully connected or because we were not able to
1588 // connect
1589 return DoWait(seconds, milliseconds,
1590 GSOCK_CONNECTION_FLAG | GSOCK_LOST_FLAG);
1591 }
1592
1593 // ==========================================================================
1594 // wxDatagramSocket
1595 // ==========================================================================
1596
1597 /* NOTE: experimental stuff - might change */
1598
1599 wxDatagramSocket::wxDatagramSocket( const wxSockAddress& addr,
1600 wxSocketFlags flags )
1601 : wxSocketBase( flags, wxSOCKET_DATAGRAM )
1602 {
1603 // Create the socket
1604 m_socket = GSocket::Create();
1605
1606 if (!m_socket)
1607 {
1608 wxFAIL_MSG( _T("datagram socket not new'd") );
1609 return;
1610 }
1611 m_socket->Notify(m_notify);
1612 // Setup the socket as non connection oriented
1613 m_socket->SetLocal(addr.GetAddress());
1614 if (flags & wxSOCKET_REUSEADDR)
1615 {
1616 m_socket->SetReusable();
1617 }
1618 if (GetFlags() & wxSOCKET_BROADCAST)
1619 {
1620 m_socket->SetBroadcast();
1621 }
1622 if (GetFlags() & wxSOCKET_NOBIND)
1623 {
1624 m_socket->DontDoBind();
1625 }
1626 if ( m_socket->SetNonOriented() != GSOCK_NOERROR )
1627 {
1628 delete m_socket;
1629 m_socket = NULL;
1630 return;
1631 }
1632
1633 // Initialize all stuff
1634 m_connected = false;
1635 m_establishing = false;
1636 m_socket->SetTimeout( m_timeout * 1000 );
1637 m_socket->SetCallback( GSOCK_INPUT_FLAG | GSOCK_OUTPUT_FLAG |
1638 GSOCK_LOST_FLAG | GSOCK_CONNECTION_FLAG,
1639 wx_socket_callback, (char*)this );
1640 }
1641
1642 wxDatagramSocket& wxDatagramSocket::RecvFrom( wxSockAddress& addr,
1643 void* buf,
1644 wxUint32 nBytes )
1645 {
1646 Read(buf, nBytes);
1647 GetPeer(addr);
1648 return (*this);
1649 }
1650
1651 wxDatagramSocket& wxDatagramSocket::SendTo( const wxSockAddress& addr,
1652 const void* buf,
1653 wxUint32 nBytes )
1654 {
1655 wxASSERT_MSG( m_socket, _T("Socket not initialised") );
1656
1657 m_socket->SetPeer(addr.GetAddress());
1658 Write(buf, nBytes);
1659 return (*this);
1660 }
1661
1662 // ==========================================================================
1663 // wxSocketModule
1664 // ==========================================================================
1665
1666 class wxSocketModule : public wxModule
1667 {
1668 public:
1669 virtual bool OnInit()
1670 {
1671 // wxSocketBase will call GSocket_Init() itself when/if needed
1672 return true;
1673 }
1674
1675 virtual void OnExit()
1676 {
1677 if ( wxSocketBase::IsInitialized() )
1678 wxSocketBase::Shutdown();
1679 }
1680
1681 private:
1682 DECLARE_DYNAMIC_CLASS(wxSocketModule)
1683 };
1684
1685 IMPLEMENT_DYNAMIC_CLASS(wxSocketModule, wxModule)
1686
1687 #endif // wxUSE_SOCKETS