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