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