1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/socket.cpp
3 // Purpose: Socket handler classes
4 // Authors: Guilhem Lavaux, Guillermo Rodriguez Garcia
6 // Copyright: (C) 1999-1997, Guilhem Lavaux
7 // (C) 2000-1999, Guillermo Rodriguez Garcia
9 // License: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ==========================================================================
14 // ==========================================================================
16 // For compilers that support precompilation, includes "wx.h".
17 #include "wx/wxprec.h"
25 #include "wx/socket.h"
28 #include "wx/object.h"
29 #include "wx/string.h"
36 #include "wx/module.h"
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"
45 // DLL options compatibility check:
47 WX_CHECK_BUILD_OPTIONS("wxNet")
49 // --------------------------------------------------------------------------
50 // macros and constants
51 // --------------------------------------------------------------------------
54 #define MAX_DISCARD_SIZE (10 * 1024)
56 #define wxTRACE_Socket _T("wxSocket")
58 // --------------------------------------------------------------------------
60 // --------------------------------------------------------------------------
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
)
68 // --------------------------------------------------------------------------
70 // --------------------------------------------------------------------------
72 class wxSocketState
: public wxObject
75 wxSocketFlags m_flags
;
76 wxSocketEventFlags m_eventmask
;
81 wxSocketState() : wxObject() {}
83 DECLARE_NO_COPY_CLASS(wxSocketState
)
86 // Conditionally make the socket non-blocking for the lifetime of this object.
87 class wxSocketUnblocker
90 wxSocketUnblocker(GSocket
*socket
, bool unblock
= true)
95 m_socket
->SetNonBlocking(true);
101 m_socket
->SetNonBlocking(false);
105 GSocket
* const m_socket
;
108 DECLARE_NO_COPY_CLASS(wxSocketUnblocker
)
111 // ============================================================================
113 // ============================================================================
115 GSocketManager
*GSocketManager::ms_manager
= NULL
;
118 void GSocketManager::Set(GSocketManager
*manager
)
120 wxASSERT_MSG( !ms_manager
, "too late to set manager now" );
122 ms_manager
= manager
;
126 void GSocketManager::Init()
128 wxASSERT_MSG( !ms_manager
, "shouldn't be initialized twice" );
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
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.
146 wxASSERT_MSG( wxIsMainThread(),
147 "sockets must be initialized from the main thread" );
149 wxAppConsole
* const app
= wxAppConsole::GetInstance();
150 wxCHECK_RET( app
, "sockets can't be initialized without wxApp" );
152 ms_manager
= app
->GetTraits()->GetSocketManager();
155 // ==========================================================================
157 // ==========================================================================
160 GSocket
*GSocketBase::Create()
162 GSocket
* const newsocket
= new GSocket();
163 if ( !GSocketManager::Get()->Init_Socket(newsocket
) )
172 GSocketBase::GSocketBase()
174 m_fd
= INVALID_SOCKET
;
178 m_error
= GSOCK_NOERROR
;
181 m_non_blocking
= false;
183 m_timeout
.tv_sec
= 10 * 60; /* 10 minutes */
184 m_timeout
.tv_usec
= 0;
186 m_timeout
= 10*60*1000; /* 10 minutes * 60 sec * 1000 millisec */
189 m_establishing
= false;
193 m_initialRecvBufferSize
= -1;
194 m_initialSendBufferSize
= -1;
196 for ( int i
= 0; i
< GSOCK_MAX_EVENT
; i
++ )
200 GSocketBase::~GSocketBase()
202 if (m_fd
!= INVALID_SOCKET
)
206 GAddress_destroy(m_local
);
209 GAddress_destroy(m_peer
);
211 // cast is ok as all GSocketBase objects we have in our code are really
213 GSocketManager::Get()->Destroy_Socket(static_cast<GSocket
*>(this));
217 * Disallow further read/write operations on this socket, close
218 * the fd and disable all callbacks.
220 void GSocketBase::Shutdown()
222 if ( m_fd
!= INVALID_SOCKET
)
224 shutdown(m_fd
, 1 /* SD_SEND */);
228 /* Disable GUI callbacks */
229 for ( int evt
= 0; evt
< GSOCK_MAX_EVENT
; evt
++ )
230 m_cbacks
[evt
] = NULL
;
232 m_detected
= GSOCK_LOST_FLAG
;
235 // ==========================================================================
237 // ==========================================================================
239 // --------------------------------------------------------------------------
240 // Initialization and shutdown
241 // --------------------------------------------------------------------------
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;
247 bool wxSocketBase::IsInitialized()
249 return m_countInit
> 0;
252 bool wxSocketBase::Initialize()
254 if ( !m_countInit
++ )
256 if ( !GSocket_Init() )
267 void wxSocketBase::Shutdown()
269 // we should be initialized
270 wxASSERT_MSG( m_countInit
> 0, _T("extra call to Shutdown()") );
271 if ( --m_countInit
== 0 )
277 // --------------------------------------------------------------------------
279 // --------------------------------------------------------------------------
281 void wxSocketBase::Init()
284 m_type
= wxSOCKET_UNINIT
;
296 m_beingDeleted
= false;
310 if ( !IsInitialized() )
312 // this Initialize() will be undone by wxSocketModule::OnExit(), all
313 // the other calls to it should be matched by a call to Shutdown()
318 wxSocketBase::wxSocketBase()
323 wxSocketBase::wxSocketBase(wxSocketFlags flags
, wxSocketType type
)
332 wxSocketBase::~wxSocketBase()
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
;
338 traits
->RemoveFromPendingDelete(this);
340 // Shutdown and close the socket
344 // Destroy the GSocket object
348 // Free the pushback buffer
353 bool wxSocketBase::Destroy()
355 // Delayed destruction: the socket will be deleted during the next idle
356 // loop iteration. This ensures that all pending events have been
358 m_beingDeleted
= true;
360 // Shutdown and close the socket
363 // Supress events from now on
366 // schedule this object for deletion
367 wxAppTraits
*traits
= wxTheApp
? wxTheApp
->GetTraits() : NULL
;
370 // let the traits object decide what to do with us
371 traits
->ScheduleForDestroy(this);
373 else // no app or no traits
375 // in wxBase we might have no app object at all, don't leak memory
382 // --------------------------------------------------------------------------
384 // --------------------------------------------------------------------------
386 // The following IO operations update m_error and m_lcount:
387 // {Read, Write, ReadMsg, WriteMsg, Peek, Unread, Discard}
389 // TODO: Should Connect, Accept and AcceptWith update m_error?
391 bool wxSocketBase::Close()
393 // Interrupt pending waits
399 m_socket
->UnsetCallback(
403 GSOCK_CONNECTION_FLAG
406 // Shutdown the connection
407 m_socket
->Shutdown();
411 m_establishing
= false;
415 wxSocketBase
& wxSocketBase::Read(void* buffer
, wxUint32 nbytes
)
420 m_lcount
= DoRead(buffer
, nbytes
);
422 // If in wxSOCKET_WAITALL mode, all bytes should have been read.
423 if (m_flags
& wxSOCKET_WAITALL
)
424 m_error
= (m_lcount
!= nbytes
);
426 m_error
= (m_lcount
== 0);
428 // Allow read events from now on
434 wxUint32
wxSocketBase::DoRead(void* buffer_
, wxUint32 nbytes
)
436 // We use pointer arithmetic here which doesn't work with void pointers.
437 char *buffer
= static_cast<char *>(buffer_
);
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
442 wxUint32 total
= GetPushback(buffer
, nbytes
, false);
446 // If it's indeed closed or if read everything, there is nothing more to do.
447 if ( !m_socket
|| !nbytes
)
450 wxCHECK_MSG( buffer
, 0, "NULL buffer" );
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
)
457 wxSocketUnblocker
unblock(m_socket
);
458 int ret
= m_socket
->Read(buffer
, nbytes
);
464 else // blocking socket
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() )
474 const int ret
= m_socket
->Read(buffer
, nbytes
);
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
487 // this will be always interpreted as error by Read()
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
496 if ( !(m_flags
& wxSOCKET_WAITALL
) )
499 // Otherwise continue reading until we do read everything.
511 wxSocketBase
& wxSocketBase::ReadMsg(void* buffer
, wxUint32 nbytes
)
513 wxUint32 len
, len2
, sig
, total
;
518 unsigned char sig
[4];
519 unsigned char len
[4];
528 SetFlags((m_flags
& wxSOCKET_BLOCK
) | wxSOCKET_WAITALL
);
530 if (DoRead(&msg
, sizeof(msg
)) != sizeof(msg
))
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);
538 if (sig
!= 0xfeeddead)
540 wxLogWarning(_("wxSocket: invalid signature in ReadMsg."));
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);
557 // Don't attempt to read if the msg was zero bytes long.
560 total
= DoRead(buffer
, len
);
568 char *discard_buffer
= new char[MAX_DISCARD_SIZE
];
571 // NOTE: discarded bytes don't add to m_lcount.
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
;
578 while ((discard_len
> 0) && len2
);
580 delete [] discard_buffer
;
585 if (DoRead(&msg
, sizeof(msg
)) != sizeof(msg
))
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);
593 if (sig
!= 0xdeadfeed)
595 wxLogWarning(_("wxSocket: invalid signature in ReadMsg."));
611 wxSocketBase
& wxSocketBase::Peek(void* buffer
, wxUint32 nbytes
)
616 m_lcount
= DoRead(buffer
, nbytes
);
617 Pushback(buffer
, m_lcount
);
619 // If in wxSOCKET_WAITALL mode, all bytes should have been read.
620 if (m_flags
& wxSOCKET_WAITALL
)
621 m_error
= (m_lcount
!= nbytes
);
623 m_error
= (m_lcount
== 0);
625 // Allow read events again
631 wxSocketBase
& wxSocketBase::Write(const void *buffer
, wxUint32 nbytes
)
636 m_lcount
= DoWrite(buffer
, nbytes
);
638 // If in wxSOCKET_WAITALL mode, all bytes should have been written.
639 if (m_flags
& wxSOCKET_WAITALL
)
640 m_error
= (m_lcount
!= nbytes
);
642 m_error
= (m_lcount
== 0);
644 // Allow write events again
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
)
654 const char *buffer
= static_cast<const char *>(buffer_
);
656 // Return if there is nothing to read or the socket is (already?) closed.
657 if ( !m_socket
|| !nbytes
)
660 wxCHECK_MSG( buffer
, 0, "NULL buffer" );
663 if ( m_flags
& wxSOCKET_NOWAIT
)
665 wxSocketUnblocker
unblock(m_socket
);
666 const int ret
= m_socket
->Write(buffer
, nbytes
);
670 else // blocking socket
674 if ( !(m_flags
& wxSOCKET_BLOCK
) && !WaitForWrite() )
677 const int ret
= m_socket
->Write(buffer
, nbytes
);
688 if ( !(m_flags
& wxSOCKET_WAITALL
) )
702 wxSocketBase
& wxSocketBase::WriteMsg(const void *buffer
, wxUint32 nbytes
)
708 unsigned char sig
[4];
709 unsigned char len
[4];
717 SetFlags((m_flags
& wxSOCKET_BLOCK
) | wxSOCKET_WAITALL
);
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;
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);
729 if (DoWrite(&msg
, sizeof(msg
)) < sizeof(msg
))
732 total
= DoWrite(buffer
, nbytes
);
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;
744 msg
.len
[3] = (char) 0;
746 if ((DoWrite(&msg
, sizeof(msg
))) < sizeof(msg
))
760 wxSocketBase
& wxSocketBase::Unread(const void *buffer
, wxUint32 nbytes
)
763 Pushback(buffer
, nbytes
);
771 wxSocketBase
& wxSocketBase::Discard()
773 char *buffer
= new char[MAX_DISCARD_SIZE
];
780 SetFlags(wxSOCKET_NOWAIT
);
784 ret
= DoRead(buffer
, MAX_DISCARD_SIZE
);
787 while (ret
== MAX_DISCARD_SIZE
);
793 // Allow read events again
799 // --------------------------------------------------------------------------
801 // --------------------------------------------------------------------------
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.
810 GSocketEventFlags
GSocketBase::Select(GSocketEventFlags flags
)
814 GSocketEventFlags result
= 0;
821 return (GSOCK_LOST_FLAG
& flags
);
823 /* Do not use a static struct, Linux can garble it */
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
);
835 /* Check 'sticky' CONNECTION flag first */
836 result
|= GSOCK_CONNECTION_FLAG
& m_detected
;
838 /* If we have already detected a LOST event, then don't try
839 * to do any further processing.
841 if ((m_detected
& GSOCK_LOST_FLAG
) != 0)
843 m_establishing
= false;
844 return (GSOCK_LOST_FLAG
& flags
);
848 if (select(m_fd
+ 1, &readfds
, &writefds
, &exceptfds
, &tv
) < 0)
850 /* What to do here? */
851 return (result
& flags
);
854 /* Check for exceptions and errors */
855 if (wxFD_ISSET(m_fd
, &exceptfds
))
857 m_establishing
= false;
858 m_detected
= GSOCK_LOST_FLAG
;
860 /* LOST event: Abort any further processing */
861 return (GSOCK_LOST_FLAG
& flags
);
864 /* Check for readability */
865 if (wxFD_ISSET(m_fd
, &readfds
))
867 result
|= GSOCK_INPUT_FLAG
;
869 if (m_server
&& m_stream
)
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
;
878 /* Check for writability */
879 if (wxFD_ISSET(m_fd
, &writefds
))
881 if (m_establishing
&& !m_server
)
884 SOCKOPTLEN_T len
= sizeof(error
);
885 m_establishing
= false;
886 getsockopt(m_fd
, SOL_SOCKET
, SO_ERROR
, (char*)&error
, &len
);
890 m_detected
= GSOCK_LOST_FLAG
;
892 /* LOST event: Abort any further processing */
893 return (GSOCK_LOST_FLAG
& flags
);
897 m_detected
|= GSOCK_CONNECTION_FLAG
;
902 result
|= GSOCK_OUTPUT_FLAG
;
906 return (result
| m_detected
) & flags
;
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.
916 wxSocketBase::DoWait(long seconds
, long milliseconds
, wxSocketEventFlags flags
)
918 wxCHECK_MSG( m_socket
, false, "can't wait on invalid socket" );
920 // This can be set to true from Interrupt() to exit this function a.s.a.p.
924 // Use either the provided timeout or the default timeout value associated
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
;
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() )
937 eventLoop
= wxEventLoop::GetActive();
939 else // in worker thread
941 // We never dispatch messages from threads other than the main one.
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;
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
);
956 // Incoming connection (server) or connection established (client)?
957 if ( result
& GSOCK_CONNECTION_FLAG
)
960 m_establishing
= false;
965 // Data available or output buffer ready?
966 if ( (result
& GSOCK_INPUT_FLAG
) || (result
& GSOCK_OUTPUT_FLAG
) )
973 if ( result
& GSOCK_LOST_FLAG
)
976 m_establishing
= false;
977 if ( flags
& GSOCK_LOST_FLAG
)
986 const wxMilliClock_t timeNow
= wxGetLocalTimeMillis();
987 if ( timeNow
>= timeEnd
)
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();
999 else // no event loop or waiting in another thread
1001 // We're busy waiting but at least give up the rest of our current
1005 #endif // wxUSE_THREADS
1011 bool wxSocketBase::Wait(long seconds
, long milliseconds
)
1013 return DoWait(seconds
, milliseconds
,
1016 GSOCK_CONNECTION_FLAG
|
1021 bool wxSocketBase::WaitForRead(long seconds
, long milliseconds
)
1023 // Check pushback buffer before entering DoWait
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
);
1035 bool wxSocketBase::WaitForWrite(long seconds
, long milliseconds
)
1037 return DoWait(seconds
, milliseconds
, GSOCK_OUTPUT_FLAG
| GSOCK_LOST_FLAG
);
1040 bool wxSocketBase::WaitForLost(long seconds
, long milliseconds
)
1042 return DoWait(seconds
, milliseconds
, GSOCK_LOST_FLAG
);
1045 // --------------------------------------------------------------------------
1047 // --------------------------------------------------------------------------
1050 // Get local or peer address
1053 bool wxSocketBase::GetPeer(wxSockAddress
& addr_man
) const
1060 peer
= m_socket
->GetPeer();
1062 // copying a null address would just trigger an assert anyway
1067 addr_man
.SetAddress(peer
);
1068 GAddress_destroy(peer
);
1073 bool wxSocketBase::GetLocal(wxSockAddress
& addr_man
) const
1080 local
= m_socket
->GetLocal();
1081 addr_man
.SetAddress(local
);
1082 GAddress_destroy(local
);
1088 // Save and restore socket state
1091 void wxSocketBase::SaveState()
1093 wxSocketState
*state
;
1095 state
= new wxSocketState();
1097 state
->m_flags
= m_flags
;
1098 state
->m_notify
= m_notify
;
1099 state
->m_eventmask
= m_eventmask
;
1100 state
->m_clientData
= m_clientData
;
1102 m_states
.Append(state
);
1105 void wxSocketBase::RestoreState()
1107 wxList::compatibility_iterator node
;
1108 wxSocketState
*state
;
1110 node
= m_states
.GetLast();
1114 state
= (wxSocketState
*)node
->GetData();
1116 m_flags
= state
->m_flags
;
1117 m_notify
= state
->m_notify
;
1118 m_eventmask
= state
->m_eventmask
;
1119 m_clientData
= state
->m_clientData
;
1121 m_states
.Erase(node
);
1126 // Timeout and flags
1129 void wxSocketBase::SetTimeout(long seconds
)
1131 m_timeout
= seconds
;
1134 m_socket
->SetTimeout(m_timeout
* 1000);
1137 void wxSocketBase::SetFlags(wxSocketFlags flags
)
1139 // Do some sanity checking on the flags used: not all values can be used
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" );
1150 // --------------------------------------------------------------------------
1152 // --------------------------------------------------------------------------
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.
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.
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.
1173 void LINKAGEMODE
wx_socket_callback(GSocket
* WXUNUSED(socket
),
1174 GSocketEvent notification
,
1177 wxSocketBase
*sckobj
= (wxSocketBase
*)cdata
;
1179 sckobj
->OnRequest((wxSocketNotify
) notification
);
1182 void wxSocketBase::OnRequest(wxSocketNotify notification
)
1184 switch(notification
)
1186 case wxSOCKET_CONNECTION
:
1187 m_establishing
= false;
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.
1195 case wxSOCKET_INPUT
:
1196 if (m_reading
|| !m_socket
->Select(GSOCK_INPUT_FLAG
))
1200 case wxSOCKET_OUTPUT
:
1201 if (m_writing
|| !m_socket
->Select(GSOCK_OUTPUT_FLAG
))
1206 m_connected
= false;
1207 m_establishing
= false;
1211 // Schedule the event
1213 wxSocketEventFlags flag
= 0;
1215 switch (notification
)
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;
1222 wxLogWarning(_("wxSocket: unknown event!."));
1226 if (((m_eventmask
& flag
) == flag
) && m_notify
)
1230 wxSocketEvent
event(m_id
);
1231 event
.m_event
= notification
;
1232 event
.m_clientData
= m_clientData
;
1233 event
.SetEventObject(this);
1235 m_handler
->AddPendingEvent(event
);
1240 void wxSocketBase::Notify(bool notify
)
1244 m_socket
->Notify(notify
);
1247 void wxSocketBase::SetNotify(wxSocketEventFlags flags
)
1249 m_eventmask
= flags
;
1252 void wxSocketBase::SetEventHandler(wxEvtHandler
& handler
, int id
)
1254 m_handler
= &handler
;
1258 // --------------------------------------------------------------------------
1260 // --------------------------------------------------------------------------
1262 void wxSocketBase::Pushback(const void *buffer
, wxUint32 size
)
1266 if (m_unread
== NULL
)
1267 m_unread
= malloc(size
);
1272 tmp
= malloc(m_unrd_size
+ size
);
1273 memcpy((char *)tmp
+ size
, m_unread
, m_unrd_size
);
1279 m_unrd_size
+= size
;
1281 memcpy(m_unread
, buffer
, size
);
1284 wxUint32
wxSocketBase::GetPushback(void *buffer
, wxUint32 size
, bool peek
)
1286 wxCHECK_MSG( buffer
, 0, "NULL buffer" );
1291 if (size
> (m_unrd_size
-m_unrd_cur
))
1292 size
= m_unrd_size
-m_unrd_cur
;
1294 memcpy(buffer
, (char *)m_unread
+ m_unrd_cur
, size
);
1299 if (m_unrd_size
== m_unrd_cur
)
1312 // ==========================================================================
1314 // ==========================================================================
1316 // --------------------------------------------------------------------------
1318 // --------------------------------------------------------------------------
1320 wxSocketServer::wxSocketServer(const wxSockAddress
& addr_man
,
1321 wxSocketFlags flags
)
1322 : wxSocketBase(flags
, wxSOCKET_SERVER
)
1324 wxLogTrace( wxTRACE_Socket
, _T("Opening wxSocketServer") );
1326 m_socket
= GSocket::Create();
1330 wxLogTrace( wxTRACE_Socket
, _T("*** GSocket_new failed") );
1334 // Setup the socket as server
1335 m_socket
->Notify(m_notify
);
1336 m_socket
->SetLocal(addr_man
.GetAddress());
1338 if (GetFlags() & wxSOCKET_REUSEADDR
) {
1339 m_socket
->SetReusable();
1341 if (GetFlags() & wxSOCKET_BROADCAST
) {
1342 m_socket
->SetBroadcast();
1344 if (GetFlags() & wxSOCKET_NOBIND
) {
1345 m_socket
->DontDoBind();
1348 if (m_socket
->SetServer() != GSOCK_NOERROR
)
1353 wxLogTrace( wxTRACE_Socket
, _T("*** GSocket_SetServer failed") );
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);
1362 wxLogTrace( wxTRACE_Socket
, _T("wxSocketServer on fd %d"), m_socket
->m_fd
);
1365 // --------------------------------------------------------------------------
1367 // --------------------------------------------------------------------------
1369 bool wxSocketServer::AcceptWith(wxSocketBase
& sock
, bool wait
)
1374 // If wait == false, then the call should be nonblocking.
1375 // When we are finished, we put the socket to blocking mode
1377 wxSocketUnblocker
unblock(m_socket
, !wait
);
1378 GSocket
* const child_socket
= m_socket
->WaitConnection();
1383 sock
.m_type
= wxSOCKET_BASE
;
1384 sock
.m_socket
= child_socket
;
1385 sock
.m_connected
= true;
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
);
1395 wxSocketBase
*wxSocketServer::Accept(bool wait
)
1397 wxSocketBase
* sock
= new wxSocketBase();
1399 sock
->SetFlags(m_flags
);
1401 if (!AcceptWith(*sock
, wait
))
1410 bool wxSocketServer::WaitForAccept(long seconds
, long milliseconds
)
1412 return DoWait(seconds
, milliseconds
, GSOCK_CONNECTION_FLAG
);
1415 bool wxSocketBase::GetOption(int level
, int optname
, void *optval
, int *optlen
)
1417 wxASSERT_MSG( m_socket
, _T("Socket not initialised") );
1419 if (m_socket
->GetSockOpt(level
, optname
, optval
, optlen
)
1427 bool wxSocketBase::SetOption(int level
, int optname
, const void *optval
,
1430 wxASSERT_MSG( m_socket
, _T("Socket not initialised") );
1432 if (m_socket
->SetSockOpt(level
, optname
, optval
, optlen
)
1440 bool wxSocketBase::SetLocal(const wxIPV4address
& local
)
1442 GAddress
* la
= local
.GetAddress();
1444 // If the address is valid, save it for use when we call Connect
1445 if (la
&& la
->m_addr
)
1447 m_localAddress
= local
;
1455 // ==========================================================================
1457 // ==========================================================================
1459 // --------------------------------------------------------------------------
1461 // --------------------------------------------------------------------------
1463 wxSocketClient::wxSocketClient(wxSocketFlags flags
)
1464 : wxSocketBase(flags
, wxSOCKET_CLIENT
)
1466 m_initialRecvBufferSize
=
1467 m_initialSendBufferSize
= -1;
1470 wxSocketClient::~wxSocketClient()
1474 // --------------------------------------------------------------------------
1476 // --------------------------------------------------------------------------
1478 bool wxSocketClient::DoConnect(const wxSockAddress
& addr_man
,
1479 const wxSockAddress
* local
,
1484 // Shutdown and destroy the socket
1489 m_socket
= GSocket::Create();
1490 m_connected
= false;
1491 m_establishing
= false;
1496 m_socket
->SetTimeout(m_timeout
* 1000);
1497 m_socket
->SetCallback(
1501 GSOCK_CONNECTION_FLAG
,
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
);
1510 // Reuse makes sense for clients too, if we are trying to rebind to the same port
1511 if (GetFlags() & wxSOCKET_REUSEADDR
)
1513 m_socket
->SetReusable();
1515 if (GetFlags() & wxSOCKET_BROADCAST
)
1517 m_socket
->SetBroadcast();
1519 if (GetFlags() & wxSOCKET_NOBIND
)
1521 m_socket
->DontDoBind();
1524 // If no local address was passed and one has been set, use the one that was Set
1525 if (!local
&& m_localAddress
.GetAddress())
1527 local
= &m_localAddress
;
1530 // Bind to the local IP address and port, when provided
1533 GAddress
* la
= local
->GetAddress();
1535 if (la
&& la
->m_addr
)
1536 m_socket
->SetLocal(la
);
1539 #if defined(__WXMSW__) || defined(__WXGTK__)
1540 m_socket
->SetInitialSocketBuffers(m_initialRecvBufferSize
, m_initialSendBufferSize
);
1543 m_socket
->SetPeer(addr_man
.GetAddress());
1544 const GSocketError err
= m_socket
->Connect(GSOCK_STREAMED
);
1546 //this will register for callbacks - must be called after m_socket->m_fd was initialized
1547 m_socket
->Notify(m_notify
);
1549 if (err
!= GSOCK_NOERROR
)
1551 if (err
== GSOCK_WOULDBLOCK
)
1552 m_establishing
= true;
1561 bool wxSocketClient::Connect(const wxSockAddress
& addr_man
, bool wait
)
1563 return DoConnect(addr_man
, NULL
, wait
);
1566 bool wxSocketClient::Connect(const wxSockAddress
& addr_man
,
1567 const wxSockAddress
& local
,
1570 return DoConnect(addr_man
, &local
, wait
);
1573 bool wxSocketClient::WaitOnConnect(long seconds
, long milliseconds
)
1577 // this happens if the initial attempt to connect succeeded without
1582 wxCHECK_MSG( m_establishing
&& m_socket
, false,
1583 "No connection establishment attempt in progress" );
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
1589 return DoWait(seconds
, milliseconds
,
1590 GSOCK_CONNECTION_FLAG
| GSOCK_LOST_FLAG
);
1593 // ==========================================================================
1595 // ==========================================================================
1597 /* NOTE: experimental stuff - might change */
1599 wxDatagramSocket::wxDatagramSocket( const wxSockAddress
& addr
,
1600 wxSocketFlags flags
)
1601 : wxSocketBase( flags
, wxSOCKET_DATAGRAM
)
1603 // Create the socket
1604 m_socket
= GSocket::Create();
1608 wxFAIL_MSG( _T("datagram socket not new'd") );
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
)
1616 m_socket
->SetReusable();
1618 if (GetFlags() & wxSOCKET_BROADCAST
)
1620 m_socket
->SetBroadcast();
1622 if (GetFlags() & wxSOCKET_NOBIND
)
1624 m_socket
->DontDoBind();
1626 if ( m_socket
->SetNonOriented() != GSOCK_NOERROR
)
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 );
1642 wxDatagramSocket
& wxDatagramSocket::RecvFrom( wxSockAddress
& addr
,
1651 wxDatagramSocket
& wxDatagramSocket::SendTo( const wxSockAddress
& addr
,
1655 wxASSERT_MSG( m_socket
, _T("Socket not initialised") );
1657 m_socket
->SetPeer(addr
.GetAddress());
1662 // ==========================================================================
1664 // ==========================================================================
1666 class wxSocketModule
: public wxModule
1669 virtual bool OnInit()
1671 // wxSocketBase will call GSocket_Init() itself when/if needed
1675 virtual void OnExit()
1677 if ( wxSocketBase::IsInitialized() )
1678 wxSocketBase::Shutdown();
1682 DECLARE_DYNAMIC_CLASS(wxSocketModule
)
1685 IMPLEMENT_DYNAMIC_CLASS(wxSocketModule
, wxModule
)
1687 #endif // wxUSE_SOCKETS